1
0

Compare commits

...

36 Commits

Author SHA1 Message Date
522b609ebc
Update description.txt 2024-04-26 23:31:48 +03:00
8223dec827
add CHN and CHT localizations
translator: cheungfatzong
https://steamcommunity.com/profiles/76561199126205919
2024-04-26 22:20:05 +03:00
e680270722
Merge pull request #7 from GenZmeY/short-name
add short alias for mutator
2024-03-08 16:45:49 +03:00
1f6de98e2f
add short alias for mutator 2023-12-31 23:13:28 +03:00
469c8bbd96
update ci/cd 2023-12-31 20:49:54 +03:00
48e1b9ff1d
update megalinter ci/cd 2023-10-21 14:12:10 +03:00
464ba0f492
Merge pull request #4 from GenZmeY/one-player
fixes for solo play
2023-06-29 02:11:19 +03:00
afadeba04c dont show notifications if there is one player on server 2023-06-29 02:09:01 +03:00
1c1b7f6e90 Merge branch 'master' into one-player 2023-06-29 01:40:44 +03:00
690aa2ba6f fix the ability to kick players who are not loaded 2023-06-29 01:39:54 +03:00
158c6b8a2e
Merge pull request #3 from GenZmeY/group-names
add mutator group
2023-06-29 01:16:16 +03:00
e354b714ca add mutator group 2023-06-03 16:53:08 +03:00
6fb5902567 update .editorconfig 2023-05-14 12:18:14 +03:00
8a26a40e90 force show up chat box 2023-05-14 02:34:29 +03:00
edb2334ac8
Merge pull request #2 from GenZmeY/MegaLinter
Mega linter
2023-05-13 02:04:52 +03:00
c37b64c3be update GPL badge 2023-05-13 02:00:55 +03:00
9635fb5007 add MegaLinter badge 2023-05-13 01:58:23 +03:00
e595daf7ca update ci/cd 2023-05-13 01:55:12 +03:00
9b47405939 Merge branch 'master' into MegaLinter 2023-05-13 01:49:29 +03:00
345e765a0e fix EOLs and remove trailing whitespaces 2023-05-13 01:48:44 +03:00
c724de802b Update README.md 2023-05-13 01:42:09 +03:00
da1f19cf9d add MegaLinter 2023-05-13 01:17:07 +03:00
777b2389c1 add .editorconfig 2023-05-13 01:16:24 +03:00
66d7dce3d0 update build tools 2023-05-13 01:14:20 +03:00
b66f2e9fef update description 2022-10-01 21:49:37 +03:00
af861b9888 update build tools 2022-09-13 05:00:53 +03:00
5f4617e25f update build tools 2022-09-02 16:21:40 +03:00
5dd4279dd4 fix player repinfo destroy 2022-09-02 15:23:34 +03:00
31db3849be update build tools 2022-09-02 15:17:59 +03:00
cdacf03d40 swap didn't voted and voted players on HUD. 2022-08-30 07:15:12 +03:00
8dfe63771b skiptrader/pause vote CD compatibility 2022-08-17 15:24:46 +03:00
97866c1a55 update description 2022-08-14 14:46:39 +03:00
2461bda5f7 update description 2022-08-14 14:44:31 +03:00
8d02c1e0a3 update description 2022-08-14 14:37:30 +03:00
3379596de4 add VoteTime setting 2022-08-14 14:17:50 +03:00
a881c1ef93 update description 2022-08-13 18:08:18 +03:00
31 changed files with 2627 additions and 2201 deletions

33
.editorconfig Normal file
View File

@ -0,0 +1,33 @@
root = true
# Global
[*]
indent_style = unset
indent_size = 4
tab_width = 4
end_of_line = lf
charset = utf-8
trim_trailing_whitespace = true
insert_final_newline = unset
# Unreal Engine 3 / Source
[*.uc]
indent_style = tab
[*.{uci,upkg}]
# Unreal Engine 3 / i18n
[*.{chn,cht,cze,dan,deu,dut,esl,esn,fra,frc,hun,int,ita,jpn,kor,pol,por,ptb,rus,tur,ukr}]
charset = utf-16le
# Other
[*.md]
indent_style = space
trim_trailing_whitespace = false
[*.yml]
indent_style = space
indent_size = 2
[*.{txt,cfg,conf}]
indent_style = tab

115
.github/workflows/mega-linter.yml vendored Normal file
View File

@ -0,0 +1,115 @@
---
name: MegaLinter
permissions: read-all
on:
push:
pull_request:
branches:
- master
env:
APPLY_FIXES: none
APPLY_FIXES_EVENT: pull_request
APPLY_FIXES_MODE: commit
FILTER_REGEX_EXCLUDE: (mega-linter.yml)
DISABLE: SPELL
DISABLE_ERRORS_LINTERS: COPYPASTE_JSCPD
concurrency:
group: ${{ github.ref }}-${{ github.workflow }}
cancel-in-progress: true
jobs:
megalinter:
name: MegaLinter
runs-on: ubuntu-latest
permissions:
contents: write
issues: write
pull-requests: write
steps:
- name: Checkout Code
uses: actions/checkout@b4ffde65f46336ab88eb53be808477a3936bae11
with:
token: ${{ secrets.PAT || secrets.GITHUB_TOKEN }}
fetch-depth: 0
- name: MegaLinter
uses: oxsecurity/megalinter@7e042c726c68415475b05a65a686c612120a1232
id: ml
env:
VALIDATE_ALL_CODEBASE: true
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
- name: Archive production artifacts
uses: actions/upload-artifact@c7d193f32edcb7bfad88892161225aeda64e9392
if: success() || failure()
with:
name: MegaLinter reports
path: |
megalinter-reports
mega-linter.log
- name: Set APPLY_FIXES_IF var
run: |
printf 'APPLY_FIXES_IF=%s\n' "${{
steps.ml.outputs.has_updated_sources == 1 &&
(
env.APPLY_FIXES_EVENT == 'all' ||
env.APPLY_FIXES_EVENT == github.event_name
) &&
(
github.event_name == 'push' ||
github.event.pull_request.head.repo.full_name == github.repository
)
}}" >> "${GITHUB_ENV}"
- name: Set APPLY_FIXES_IF_* vars
run: |
printf 'APPLY_FIXES_IF_PR=%s\n' "${{
env.APPLY_FIXES_IF == 'true' &&
env.APPLY_FIXES_MODE == 'pull_request'
}}" >> "${GITHUB_ENV}"
printf 'APPLY_FIXES_IF_COMMIT=%s\n' "${{
env.APPLY_FIXES_IF == 'true' &&
env.APPLY_FIXES_MODE == 'commit' &&
(!contains(fromJSON('["refs/heads/main", "refs/heads/master"]'), github.ref))
}}" >> "${GITHUB_ENV}"
- name: Create Pull Request with applied fixes
uses: peter-evans/create-pull-request@153407881ec5c347639a548ade7d8ad1d6740e38
id: cpr
if: env.APPLY_FIXES_IF_PR == 'true'
with:
token: ${{ secrets.PAT || secrets.GITHUB_TOKEN }}
commit-message: "[MegaLinter] Apply linters automatic fixes"
title: "[MegaLinter] Apply linters automatic fixes"
labels: bot
- name: Create PR output
if: env.APPLY_FIXES_IF_PR == 'true'
run: |
echo "PR Number - ${{ steps.cpr.outputs.pull-request-number }}"
echo "PR URL - ${{ steps.cpr.outputs.pull-request-url }}"
- name: Prepare commit
if: env.APPLY_FIXES_IF_COMMIT == 'true'
run: sudo chown -Rc $UID .git/
- name: Commit and push applied linter fixes
uses: stefanzweifel/git-auto-commit-action@8756aa072ef5b4a080af5dc8fef36c5d586e521d
if: env.APPLY_FIXES_IF_COMMIT == 'true'
with:
branch: >-
${{
github.event.pull_request.head.ref ||
github.head_ref ||
github.ref
}}
commit_message: "[MegaLinter] Apply linters fixes"
commit_user_name: "github-actions"
commit_user_email: "github-actions[bot]@users.noreply.github.com"

2
.gitignore vendored Normal file
View File

@ -0,0 +1,2 @@
*.psd
/ignore

View File

@ -1,103 +1,114 @@
class BaseVote extends Object class BaseVote extends Object
config(CVC) config(CVC)
abstract; abstract;
var public config String PositiveColorHex; var public config String PositiveColorHex;
var public config String NegativeColorHex; var public config String NegativeColorHex;
var public config bool bChatNotifications; var public config bool bChatNotifications;
var public config bool bHudNotifications; var public config bool bHudNotifications;
var public config float DefferedClearHUD; var public config float DefferedClearHUD;
var public config int VoteTime;
public static function InitConfig(int Version, int LatestVersion, E_LogLevel LogLevel)
{ public static function InitConfig(int Version, int LatestVersion, E_LogLevel LogLevel)
`Log_TraceStatic(); {
`Log_TraceStatic();
switch (Version)
{ switch (Version)
case `NO_CONFIG: {
ApplyDefault(LogLevel); case `NO_CONFIG:
ApplyDefault(LogLevel);
default: break;
} case 1:
default.VoteTime = class'KFVoteCollector'.default.VoteTime;
if (LatestVersion != Version)
{ default: break;
StaticSaveConfig(); }
}
} if (LatestVersion != Version)
{
public static function Load(E_LogLevel LogLevel) StaticSaveConfig();
{ }
`Log_TraceStatic(); }
if (!IsValidHexColor(default.PositiveColorHex, LogLevel)) public static function Load(E_LogLevel LogLevel)
{ {
`Log_Error("PositiveColorHex" @ "(" $ default.PositiveColorHex $ ")" @ "is not valid hex color"); `Log_TraceStatic();
default.PositiveColorHex = class'KFLocalMessage'.default.EventColor;
} if (!IsValidHexColor(default.PositiveColorHex, LogLevel))
{
if (!IsValidHexColor(default.NegativeColorHex, LogLevel)) `Log_Error("PositiveColorHex" @ "(" $ default.PositiveColorHex $ ")" @ "is not valid hex color");
{ default.PositiveColorHex = class'KFLocalMessage'.default.EventColor;
`Log_Error("NegativeColorHex" @ "(" $ default.NegativeColorHex $ ")" @ "is not valid hex color"); }
default.NegativeColorHex = class'KFLocalMessage'.default.InteractionColor;
} if (!IsValidHexColor(default.NegativeColorHex, LogLevel))
{
if (default.DefferedClearHUD < 0) `Log_Error("NegativeColorHex" @ "(" $ default.NegativeColorHex $ ")" @ "is not valid hex color");
{ default.NegativeColorHex = class'KFLocalMessage'.default.InteractionColor;
`Log_Error("DefferedClearHUD" @ "(" $ default.DefferedClearHUD $ ")" @ "must be greater than 0"); }
default.DefferedClearHUD = 0.0f;
} if (default.DefferedClearHUD < 0)
} {
`Log_Error("DefferedClearHUD" @ "(" $ default.DefferedClearHUD $ ")" @ "must be greater than 0");
protected static function ApplyDefault(E_LogLevel LogLevel) default.DefferedClearHUD = 0.0f;
{ }
`Log_TraceStatic();
if (default.VoteTime <= 0 || default.VoteTime > 255)
default.bChatNotifications = true; {
default.bHudNotifications = true; `Log_Error("VoteTime" @ "(" $ default.VoteTime $ ")" @ "must be in range 1-255");
default.PositiveColorHex = class'KFLocalMessage'.default.EventColor; default.VoteTime = class'KFVoteCollector'.default.VoteTime;
default.NegativeColorHex = class'KFLocalMessage'.default.InteractionColor; }
default.DefferedClearHUD = 1.0f; }
}
protected static function ApplyDefault(E_LogLevel LogLevel)
protected static function bool IsValidHexColor(String HexColor, E_LogLevel LogLevel) {
{ `Log_TraceStatic();
local byte Index;
default.bChatNotifications = true;
`Log_TraceStatic(); default.bHudNotifications = true;
default.PositiveColorHex = class'KFLocalMessage'.default.EventColor;
if (len(HexColor) != 6) return false; default.NegativeColorHex = class'KFLocalMessage'.default.InteractionColor;
default.DefferedClearHUD = 1.0f;
HexColor = Locs(HexColor); default.VoteTime = class'KFVoteCollector'.default.VoteTime;
}
for (Index = 0; Index < 6; ++Index)
{ protected static function bool IsValidHexColor(String HexColor, E_LogLevel LogLevel)
switch (Mid(HexColor, Index, 1)) {
{ local byte Index;
case "0": break;
case "1": break; `Log_TraceStatic();
case "2": break;
case "3": break; if (len(HexColor) != 6) return false;
case "4": break;
case "5": break; HexColor = Locs(HexColor);
case "6": break;
case "7": break; for (Index = 0; Index < 6; ++Index)
case "8": break; {
case "9": break; switch (Mid(HexColor, Index, 1))
case "a": break; {
case "b": break; case "0": break;
case "c": break; case "1": break;
case "d": break; case "2": break;
case "e": break; case "3": break;
case "f": break; case "4": break;
default: return false; case "5": break;
} case "6": break;
} case "7": break;
case "8": break;
return true; case "9": break;
} case "a": break;
case "b": break;
defaultproperties case "c": break;
{ case "d": break;
case "e": break;
} case "f": break;
default: return false;
}
}
return true;
}
defaultproperties
{
}

View File

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

View File

@ -1,4 +1,4 @@
[Flags] [Flags]
AllowDownload=True AllowDownload=True
ClientOptional=False ClientOptional=False
ServerSideOnly=False ServerSideOnly=False

View File

@ -1,62 +1 @@
class CVCMut extends KFMutator; class CVCMut extends Mut; // backward compatibility
var private CVC CVC;
public simulated function bool SafeDestroy()
{
return (bPendingDelete || bDeleteMe || Destroy());
}
public event PreBeginPlay()
{
Super.PreBeginPlay();
if (WorldInfo.NetMode == NM_Client) return;
foreach WorldInfo.DynamicActors(class'CVC', CVC)
{
`Log_Base("Found 'CVC'");
break;
}
if (CVC == None)
{
`Log_Base("Spawn 'CVC'");
CVC = WorldInfo.Spawn(class'CVC');
}
if (CVC == None)
{
`Log_Base("Can't Spawn 'CVC', Destroy...");
SafeDestroy();
}
}
public function AddMutator(Mutator Mut)
{
if (Mut == Self) return;
if (Mut.Class == Class)
Mut.Destroy();
else
Super.AddMutator(Mut);
}
public function NotifyLogin(Controller C)
{
Super.NotifyLogin(C);
CVC.NotifyLogin(C);
}
public function NotifyLogout(Controller C)
{
Super.NotifyLogout(C);
CVC.NotifyLogout(C);
}
DefaultProperties
{
}

View File

@ -1,157 +1,153 @@
class CVC_LocalMessage extends Object class CVC_LocalMessage extends Object
abstract; abstract;
var const String PlayerIsKickProtectedDefault; var const String PlayerIsKickProtectedDefault;
var private localized String PlayerIsKickProtected; var private localized String PlayerIsKickProtected;
var const String PlayerIsStartWaveKickProtectedDefault; var const String PlayerIsStartWaveKickProtectedDefault;
var private localized String PlayerIsStartWaveKickProtected; var private localized String PlayerIsStartWaveKickProtected;
var const String PlayerCantStartKickVoteDefault; var const String PlayerCantStartKickVoteDefault;
var private localized String PlayerCantStartKickVote; var private localized String PlayerCantStartKickVote;
var const String KickVoteNotEnoughPlayersDefault; var const String PlayerPerkIsNotLoadedDefault;
var private localized String KickVoteNotEnoughPlayers; var private localized String PlayerPerkIsNotLoaded;
var const String KickVoteStartedDefault; var const String KickVoteNotEnoughPlayersDefault;
var private localized String KickVoteStarted; var private localized String KickVoteNotEnoughPlayers;
var const String KickVoteStartedForPlayerDefault; var const String KickVoteStartedDefault;
var private localized String KickVoteStartedForPlayer; var private localized String KickVoteStarted;
var const String KickVoteNotStartedForPlayerDefault; var const String KickVoteStartedForPlayerDefault;
var private localized String KickVoteNotStartedForPlayer; var private localized String KickVoteStartedForPlayer;
var const String VotedPlayersDefault; var const String KickVoteNotStartedForPlayerDefault;
var private localized String VotedPlayers; var private localized String KickVoteNotStartedForPlayer;
var const String DidntVotePlayersDefault; var const String VotedPlayersDefault;
var private localized String DidntVotePlayers; var private localized String VotedPlayers;
// TODO: var const String DidntVotePlayersDefault;
/* var private localized String DidntVotePlayers;
Kick vote hud:
start vote + only yes votes: enum E_CVC_LocalMessageType
header: <player vote for player> {
second line: yes votes CVC_PlayerIsKickProtected,
CVC_PlayerIsStartWaveKickProtected,
pause and skip: CVC_PlayerCantStartKickVote,
first line: voted players CVC_PlayerPerkIsNotLoaded,
second line: dont voted players
*/ CVC_KickVoteNotEnoughPlayers,
CVC_KickVoteStarted,
enum E_CVC_LocalMessageType CVC_KickVoteStartedForPlayer,
{ CVC_KickVoteNotStartedForPlayer,
CVC_PlayerIsKickProtected,
CVC_PlayerIsStartWaveKickProtected, CVC_KickVoteYesReceived,
CVC_PlayerCantStartKickVote, CVC_KickVoteNoReceived,
CVC_KickVoteStartedHUD,
CVC_KickVoteNotEnoughPlayers, CVC_KickVoteReceivedHUD,
CVC_KickVoteStarted,
CVC_KickVoteStartedForPlayer, CVC_SkipVoteYesReceived,
CVC_KickVoteNotStartedForPlayer, CVC_SkipVoteNoReceived,
CVC_KickVoteYesReceived, CVC_PauseVoteYesReceived,
CVC_KickVoteNoReceived, CVC_PauseVoteNoReceived,
CVC_KickVoteStartedHUD,
CVC_KickVoteReceivedHUD, CVC_VoteProgressHUD,
};
CVC_SkipVoteYesReceived,
CVC_SkipVoteNoReceived, private static function String ReplKicker(String Str, String Kicker)
{
CVC_PauseVoteYesReceived, return Repl(Str, "<kicker>", Kicker, false);
CVC_PauseVoteNoReceived, }
CVC_VoteProgressHUD, private static function String ReplKickee(String Str, String Kickee)
}; {
return Repl(Str, "<kickee>", Kickee, false);
private static function String ReplKicker(String Str, String Kicker) }
{
return Repl(Str, "<kicker>", Kicker, false); private static function String ReplWaves(String Str, String Waves)
} {
return Repl(Str, "<waves>", Waves, false);
private static function String ReplKickee(String Str, String Kickee) }
{
return Repl(Str, "<kickee>", Kickee, false); public static function String GetLocalizedString(
} E_LogLevel LogLevel,
E_CVC_LocalMessageType LMT,
private static function String ReplWaves(String Str, String Waves) optional String String1,
{ optional String String2,
return Repl(Str, "<waves>", Waves, false); optional String String3)
} {
`Log_TraceStatic();
public static function String GetLocalizedString(
E_LogLevel LogLevel, switch (LMT)
E_CVC_LocalMessageType LMT, {
optional String String1, case CVC_PlayerIsKickProtected:
optional String String2, return ReplKickee(default.PlayerIsKickProtected != "" ? default.PlayerIsKickProtected : default.PlayerIsKickProtectedDefault, String1);
optional String String3)
{ case CVC_PlayerIsStartWaveKickProtected:
`Log_TraceStatic(); return ReplWaves(ReplKickee(default.PlayerIsStartWaveKickProtected != "" ? default.PlayerIsStartWaveKickProtected : default.PlayerIsStartWaveKickProtectedDefault, String1), String2);
switch (LMT) case CVC_PlayerCantStartKickVote:
{ return ReplWaves(default.PlayerCantStartKickVote != "" ? default.PlayerCantStartKickVote : default.PlayerCantStartKickVoteDefault, String1);
case CVC_PlayerIsKickProtected:
return ReplKickee(default.PlayerIsKickProtected != "" ? default.PlayerIsKickProtected : default.PlayerIsKickProtectedDefault, String1); case CVC_PlayerPerkIsNotLoaded:
return ReplKickee(default.PlayerPerkIsNotLoaded != "" ? default.PlayerPerkIsNotLoaded : default.PlayerPerkIsNotLoadedDefault, String1);
case CVC_PlayerIsStartWaveKickProtected:
return ReplWaves(ReplKickee(default.PlayerIsStartWaveKickProtected != "" ? default.PlayerIsStartWaveKickProtected : default.PlayerIsStartWaveKickProtectedDefault, String1), String2); case CVC_KickVoteNotEnoughPlayers:
return ReplWaves(default.KickVoteNotEnoughPlayers != "" ? default.KickVoteNotEnoughPlayers : default.KickVoteNotEnoughPlayersDefault, String1);
case CVC_PlayerCantStartKickVote:
return ReplWaves(default.PlayerCantStartKickVote != "" ? default.PlayerCantStartKickVote : default.PlayerCantStartKickVoteDefault, String1); case CVC_KickVoteYesReceived:
return (String1 $ ":" @ class'KFCommon_LocalizedStrings'.default.YesString);
case CVC_KickVoteNotEnoughPlayers:
return ReplWaves(default.KickVoteNotEnoughPlayers != "" ? default.KickVoteNotEnoughPlayers : default.KickVoteNotEnoughPlayersDefault, String1); case CVC_KickVoteNoReceived:
return (String1 $ ":" @ class'KFCommon_LocalizedStrings'.default.NoString);
case CVC_KickVoteYesReceived:
return (String1 $ ":" @ class'KFCommon_LocalizedStrings'.default.YesString); case CVC_KickVoteStartedHUD:
return ReplKickee(ReplKicker((default.KickVoteStarted != "" ? default.KickVoteStarted : default.KickVoteStartedDefault), String1), String2) $ "\n" $ class'KFCommon_LocalizedStrings'.default.YesString $ ":" @ String3;
case CVC_KickVoteNoReceived:
return (String1 $ ":" @ class'KFCommon_LocalizedStrings'.default.NoString); case CVC_KickVoteReceivedHUD:
return class'KFCommon_LocalizedStrings'.default.YesString $ ":" @ String1 $ "\n" $ class'KFCommon_LocalizedStrings'.default.NoString $ ":" @ String2;
case CVC_KickVoteStartedHUD:
return ReplKickee(ReplKicker((default.KickVoteStarted != "" ? default.KickVoteStarted : default.KickVoteStartedDefault), String1), String2) $ "\n" $ class'KFCommon_LocalizedStrings'.default.YesString $ ":" @ String3; case CVC_KickVoteStarted:
return ReplKickee(ReplKicker((default.KickVoteStarted != "" ? default.KickVoteStarted : default.KickVoteStartedDefault), String1), String2);
case CVC_KickVoteReceivedHUD:
return class'KFCommon_LocalizedStrings'.default.YesString $ ":" @ String1 $ "\n" $ class'KFCommon_LocalizedStrings'.default.NoString $ ":" @ String2; case CVC_KickVoteStartedForPlayer:
return ReplKicker((default.KickVoteStartedForPlayer != "" ? default.KickVoteStartedForPlayer : default.KickVoteStartedForPlayerDefault), String1);
case CVC_KickVoteStarted:
return ReplKickee(ReplKicker((default.KickVoteStarted != "" ? default.KickVoteStarted : default.KickVoteStartedDefault), String1), String2); case CVC_KickVoteNotStartedForPlayer:
return ReplKicker((default.KickVoteNotStartedForPlayer != "" ? default.KickVoteNotStartedForPlayer : default.KickVoteNotStartedForPlayerDefault), String1);
case CVC_KickVoteStartedForPlayer:
return ReplKicker((default.KickVoteStartedForPlayer != "" ? default.KickVoteStartedForPlayer : default.KickVoteStartedForPlayerDefault), String1); case CVC_SkipVoteYesReceived:
return (String1 $ ":" @ class'KFCommon_LocalizedStrings'.default.YesString);
case CVC_KickVoteNotStartedForPlayer:
return ReplKicker((default.KickVoteNotStartedForPlayer != "" ? default.KickVoteNotStartedForPlayer : default.KickVoteNotStartedForPlayerDefault), String1); case CVC_SkipVoteNoReceived:
return (String1 $ ":" @ class'KFCommon_LocalizedStrings'.default.NoString);
case CVC_SkipVoteYesReceived:
return (String1 $ ":" @ class'KFCommon_LocalizedStrings'.default.YesString); case CVC_PauseVoteYesReceived:
return (String1 $ ":" @ class'KFCommon_LocalizedStrings'.default.YesString);
case CVC_SkipVoteNoReceived:
return (String1 $ ":" @ class'KFCommon_LocalizedStrings'.default.NoString); case CVC_PauseVoteNoReceived:
return (String1 $ ":" @ class'KFCommon_LocalizedStrings'.default.NoString);
case CVC_PauseVoteYesReceived:
return (String1 $ ":" @ class'KFCommon_LocalizedStrings'.default.YesString); case CVC_VoteProgressHUD:
return (default.DidntVotePlayers != "" ? default.DidntVotePlayers : default.DidntVotePlayersDefault) @ String2 $ (String1 != "" ? ("\n" $ (default.VotedPlayers != "" ? default.VotedPlayers : default.VotedPlayersDefault) @ String1) : "");
case CVC_PauseVoteNoReceived: }
return (String1 $ ":" @ class'KFCommon_LocalizedStrings'.default.NoString);
return "";
case CVC_VoteProgressHUD: }
return (default.VotedPlayers != "" ? default.VotedPlayers : default.VotedPlayersDefault) @ String1 $ (String2 != "" ? ("\n" $ (default.DidntVotePlayers != "" ? default.DidntVotePlayers : default.DidntVotePlayersDefault) @ String2) : "");
} defaultproperties
{
return ""; PlayerIsKickProtectedDefault = "<kickee> is protected from kick"
} PlayerIsStartWaveKickProtectedDefault = "You can't kick <kickee> right now. He can be kicked when he plays at least <waves> wave(s)"
PlayerCantStartKickVoteDefault = "You can't start kick vote now. You can start kick vote when you play at least <waves> wave(s)"
defaultproperties PlayerPerkIsNotLoadedDefault = "You can't kick a player who hasn't loaded yet (<kickee>)"
{ KickVoteNotEnoughPlayersDefault = "Not enough players to start vote (only players who have played at least <waves> wave(s) can vote)"
PlayerIsKickProtectedDefault = "<kickee> is protected from kick" KickVoteStartedDefault = "<kicker> has started a vote to kick <kickee>"
PlayerIsStartWaveKickProtectedDefault = "You can't kick <kickee> right now. He can be kicked when he plays at least <waves> wave(s)" KickVoteStartedForPlayerDefault = "<kicker> started voting to kick you"
PlayerCantStartKickVoteDefault = "You can't start kick vote now. You can start kick vote when you play at least <waves> wave(s)" KickVoteNotStartedForPlayerDefault = "<kicker> tried to kick you"
KickVoteNotEnoughPlayersDefault = "Not enough players to start vote (only players who have played at least <waves> wave(s) can vote)" VotedPlayersDefault = "Voted:"
KickVoteStartedDefault = "<kicker> has started a vote to kick <kickee>" DidntVotePlayersDefault = "Didn't vote:"
KickVoteStartedForPlayerDefault = "<kicker> started voting to kick you"
KickVoteNotStartedForPlayerDefault = "<kicker> tried to kick you"
VotedPlayersDefault = "Voted:"
DidntVotePlayersDefault = "Didn't vote:"
} }

View File

@ -1,123 +1,124 @@
class CVC_RepInfo extends ReplicationInfo; class CVC_RepInfo extends ReplicationInfo;
const CVCLMT = class'CVC_LocalMessage'; const CVCLMT = class'CVC_LocalMessage';
var public E_LogLevel LogLevel; var public E_LogLevel LogLevel;
var public CVC CVC; var public CVC CVC;
var public int StartWave; var public int StartWave;
var private KFPlayerController KFPC; var private KFPlayerController KFPC;
replication replication
{ {
if (bNetInitial && Role == ROLE_Authority) if (bNetInitial && Role == ROLE_Authority)
LogLevel; LogLevel;
} }
public simulated function bool SafeDestroy() public simulated function bool SafeDestroy()
{ {
`Log_Trace(); `Log_Trace();
return (bPendingDelete || bDeleteMe || Destroy()); return (bPendingDelete || bDeleteMe || Destroy());
} }
public reliable client function WriteToChatLocalized(E_CVC_LocalMessageType LMT, optional String HexColor, optional String String1, optional String String2, optional String String3) public reliable client function WriteToChatLocalized(E_CVC_LocalMessageType LMT, optional String HexColor, optional String String1, optional String String2, optional String String3)
{ {
`Log_Trace(); `Log_Trace();
WriteToChat(CVCLMT.static.GetLocalizedString(LogLevel, LMT, String1, String2, String3), HexColor); WriteToChat(CVCLMT.static.GetLocalizedString(LogLevel, LMT, String1, String2, String3), HexColor);
} }
public reliable client function WriteToChat(String Message, optional String HexColor) public reliable client function WriteToChat(String Message, optional String HexColor)
{ {
local KFGFxHudWrapper HUD; local KFGFxHudWrapper HUD;
`Log_Trace(); `Log_Trace();
if (GetKFPC() == None) return; if (GetKFPC() == None) return;
if (KFPC.MyGFxManager.PartyWidget != None && KFPC.MyGFxManager.PartyWidget.PartyChatWidget != None) if (KFPC.MyGFxManager.PartyWidget != None && KFPC.MyGFxManager.PartyWidget.PartyChatWidget != None)
{ {
KFPC.MyGFxManager.PartyWidget.PartyChatWidget.AddChatMessage(Message, HexColor); KFPC.MyGFxManager.PartyWidget.PartyChatWidget.SetVisible(true);
} KFPC.MyGFxManager.PartyWidget.PartyChatWidget.AddChatMessage(Message, HexColor);
}
HUD = KFGFxHudWrapper(KFPC.myHUD);
if (HUD != None && HUD.HUDMovie != None && HUD.HUDMovie.HudChatBox != None) HUD = KFGFxHudWrapper(KFPC.myHUD);
{ if (HUD != None && HUD.HUDMovie != None && HUD.HUDMovie.HudChatBox != None)
HUD.HUDMovie.HudChatBox.AddChatMessage(Message, HexColor); {
} HUD.HUDMovie.HudChatBox.AddChatMessage(Message, HexColor);
} }
}
public reliable client function WriteToHUDLocalized(E_CVC_LocalMessageType LMT, optional String String1, optional String String2, optional String String3, optional float DisplayTime = 0.0f)
{ public reliable client function WriteToHUDLocalized(E_CVC_LocalMessageType LMT, optional String String1, optional String String2, optional String String3, optional float DisplayTime = 0.0f)
`Log_Trace(); {
`Log_Trace();
WriteToHUD(CVCLMT.static.GetLocalizedString(LogLevel, LMT, String1, String2, String3), DisplayTime);
} WriteToHUD(CVCLMT.static.GetLocalizedString(LogLevel, LMT, String1, String2, String3), DisplayTime);
}
public reliable client function WriteToHUD(String Message, optional float DisplayTime = 0.0f)
{ public reliable client function WriteToHUD(String Message, optional float DisplayTime = 0.0f)
`Log_Trace(); {
`Log_Trace();
if (GetKFPC() == None) return;
if (GetKFPC() == None) return;
if (DisplayTime == 0.0f)
{ if (DisplayTime == 0.0f)
DisplayTime = CalcDisplayTime(Message); {
} DisplayTime = CalcDisplayTime(Message);
}
if (KFPC.MyGFxHUD != None)
{ if (KFPC.MyGFxHUD != None)
KFPC.MyGFxHUD.DisplayMapText(Message, DisplayTime, false); {
} KFPC.MyGFxHUD.DisplayMapText(Message, DisplayTime, false);
} }
}
public reliable client function DefferedClearMessageHUD(optional float Time = 0.0f)
{ public reliable client function DefferedClearMessageHUD(optional float Time = 0.0f)
`Log_Trace(); {
`Log_Trace();
SetTimer(Time, false, nameof(ClearMessageHUD));
} SetTimer(Time, false, nameof(ClearMessageHUD));
}
public reliable client function ClearMessageHUD()
{ public reliable client function ClearMessageHUD()
`Log_Trace(); {
`Log_Trace();
if (GetKFPC() == None) return;
if (GetKFPC() == None) return;
if (KFPC.MyGFxHUD != None && KFPC.MyGFxHUD.MapTextWidget != None)
{ if (KFPC.MyGFxHUD != None && KFPC.MyGFxHUD.MapTextWidget != None)
KFPC.MyGFxHUD.MapTextWidget.StoredMessageList.Length = 0; {
KFPC.MyGFxHUD.MapTextWidget.HideMessage(); KFPC.MyGFxHUD.MapTextWidget.StoredMessageList.Length = 0;
} KFPC.MyGFxHUD.MapTextWidget.HideMessage();
} }
}
private function float CalcDisplayTime(String Message)
{ private function float CalcDisplayTime(String Message)
`Log_Trace(); {
`Log_Trace();
return FClamp(Len(Message) / 20.0f, 3, 30);
} return FClamp(Len(Message) / 20.0f, 3, 30);
}
public simulated function KFPlayerController GetKFPC()
{ public simulated function KFPlayerController GetKFPC()
`Log_Trace(); {
`Log_Trace();
if (KFPC != None) return KFPC;
if (KFPC != None) return KFPC;
KFPC = KFPlayerController(Owner);
KFPC = KFPlayerController(Owner);
if (KFPC == None && ROLE < ROLE_Authority)
{ if (KFPC == None && ROLE < ROLE_Authority)
KFPC = KFPlayerController(GetALocalPlayerController()); {
} KFPC = KFPlayerController(GetALocalPlayerController());
}
return KFPC;
} return KFPC;
}
defaultproperties
{ defaultproperties
bAlwaysRelevant = false {
bOnlyRelevantToOwner = true bAlwaysRelevant = false
bSkipActorPropertyReplication = false bOnlyRelevantToOwner = true
} bSkipActorPropertyReplication = false
}

File diff suppressed because it is too large Load Diff

View File

@ -1,82 +1,82 @@
class KickProtected extends Object class KickProtected extends Object
config(CVC) config(CVC)
abstract; abstract;
var public config bool NotifyPlayerAboutKickAttempt; var public config bool NotifyPlayerAboutKickAttempt;
var private config Array<String> PlayerID; var private config Array<String> PlayerID;
public static function InitConfig(int Version, int LatestVersion, E_LogLevel LogLevel) public static function InitConfig(int Version, int LatestVersion, E_LogLevel LogLevel)
{ {
`Log_TraceStatic(); `Log_TraceStatic();
switch (Version) switch (Version)
{ {
case `NO_CONFIG: case `NO_CONFIG:
ApplyDefault(LogLevel); ApplyDefault(LogLevel);
default: break; default: break;
} }
if (LatestVersion != Version) if (LatestVersion != Version)
{ {
StaticSaveConfig(); StaticSaveConfig();
} }
} }
public static function Array<UniqueNetId> Load(E_LogLevel LogLevel) public static function Array<UniqueNetId> Load(E_LogLevel LogLevel)
{ {
local Array<UniqueNetId> UIDs; local Array<UniqueNetId> UIDs;
local UniqueNetId UID; local UniqueNetId UID;
local String ID; local String ID;
`Log_TraceStatic(); `Log_TraceStatic();
foreach default.PlayerID(ID) foreach default.PlayerID(ID)
{ {
if (AnyToUID(ID, UID, LogLevel)) if (AnyToUID(ID, UID, LogLevel))
{ {
UIDs.AddItem(UID); UIDs.AddItem(UID);
} }
else else
{ {
`Log_Warn("Can't load PlayerID:" @ ID); `Log_Warn("Can't load PlayerID:" @ ID);
} }
} }
return UIDs; return UIDs;
} }
private static function ApplyDefault(E_LogLevel LogLevel) private static function ApplyDefault(E_LogLevel LogLevel)
{ {
`Log_TraceStatic(); `Log_TraceStatic();
default.NotifyPlayerAboutKickAttempt = true; default.NotifyPlayerAboutKickAttempt = true;
default.PlayerID.Length = 0; default.PlayerID.Length = 0;
default.PlayerID.AddItem("76561190000000000"); default.PlayerID.AddItem("76561190000000000");
default.PlayerID.AddItem("0x0000000000000000"); default.PlayerID.AddItem("0x0000000000000000");
} }
private static function bool IsUID(String ID, E_LogLevel LogLevel) private static function bool IsUID(String ID, E_LogLevel LogLevel)
{ {
`Log_TraceStatic(); `Log_TraceStatic();
return (Locs(Left(ID, 2)) ~= "0x"); return (Left(ID, 2) ~= "0x");
} }
private static function bool AnyToUID(String ID, out UniqueNetId UID, E_LogLevel LogLevel) private static function bool AnyToUID(String ID, out UniqueNetId UID, E_LogLevel LogLevel)
{ {
local OnlineSubsystem OS; local OnlineSubsystem OS;
`Log_TraceStatic(); `Log_TraceStatic();
OS = class'GameEngine'.static.GetOnlineSubsystem(); OS = class'GameEngine'.static.GetOnlineSubsystem();
if (OS == None) return false; if (OS == None) return false;
return IsUID(ID, LogLevel) ? OS.StringToUniqueNetId(ID, UID) : OS.Int64ToUniqueNetId(ID, UID); return IsUID(ID, LogLevel) ? OS.StringToUniqueNetId(ID, UID) : OS.Int64ToUniqueNetId(ID, UID);
} }
defaultproperties defaultproperties
{ {
} }

View File

@ -1,52 +1,52 @@
class KickVote extends BaseVote class KickVote extends BaseVote
config(CVC) config(CVC)
abstract; abstract;
var public config bool bHudNotificationsOnlyOnTraderTime; var public config bool bHudNotificationsOnlyOnTraderTime;
var public config int MinVotingPlayersToStartKickVote; var public config int MinVotingPlayersToStartKickVote;
var public config int MaxKicks; var public config int MaxKicks;
var public config bool bLogKickVote; var public config bool bLogKickVote;
var public config String WarningColorHex; var public config String WarningColorHex;
public static function Load(E_LogLevel LogLevel) public static function Load(E_LogLevel LogLevel)
{ {
`Log_TraceStatic(); `Log_TraceStatic();
if (default.MinVotingPlayersToStartKickVote < 2) if (default.MinVotingPlayersToStartKickVote < 2)
{ {
`Log_Error("MinVotingPlayersToStartKickVote" @ "(" $ default.MinVotingPlayersToStartKickVote $ ")" @ "must be greater than 1"); `Log_Error("MinVotingPlayersToStartKickVote" @ "(" $ default.MinVotingPlayersToStartKickVote $ ")" @ "must be greater than 1");
default.MinVotingPlayersToStartKickVote = 2; default.MinVotingPlayersToStartKickVote = 2;
} }
if (default.MaxKicks < 1) if (default.MaxKicks < 1)
{ {
`Log_Error("MaxKicks" @ "(" $ default.MaxKicks $ ")" @ "must be greater than 0"); `Log_Error("MaxKicks" @ "(" $ default.MaxKicks $ ")" @ "must be greater than 0");
default.MaxKicks = 2; default.MaxKicks = 2;
} }
if (!IsValidHexColor(default.WarningColorHex, LogLevel)) if (!IsValidHexColor(default.WarningColorHex, LogLevel))
{ {
`Log_Error("WarningColorHex" @ "(" $ default.WarningColorHex $ ")" @ "is not valid hex color"); `Log_Error("WarningColorHex" @ "(" $ default.WarningColorHex $ ")" @ "is not valid hex color");
default.WarningColorHex = class'KFLocalMessage'.default.PriorityColor; default.WarningColorHex = class'KFLocalMessage'.default.PriorityColor;
} }
Super.Load(LogLevel); Super.Load(LogLevel);
} }
protected static function ApplyDefault(E_LogLevel LogLevel) protected static function ApplyDefault(E_LogLevel LogLevel)
{ {
`Log_TraceStatic(); `Log_TraceStatic();
Super.ApplyDefault(LogLevel); Super.ApplyDefault(LogLevel);
default.bHudNotificationsOnlyOnTraderTime = true; default.bHudNotificationsOnlyOnTraderTime = true;
default.MinVotingPlayersToStartKickVote = 2; default.MinVotingPlayersToStartKickVote = 2;
default.MaxKicks = 2; default.MaxKicks = 2;
default.DefferedClearHUD = 2.0f; default.DefferedClearHUD = 2.0f;
default.WarningColorHex = class'KFLocalMessage'.default.PriorityColor; default.WarningColorHex = class'KFLocalMessage'.default.PriorityColor;
} }
defaultproperties defaultproperties
{ {
} }

View File

@ -1,57 +1,57 @@
class MapStat extends Object class MapStat extends Object
config(CVC) config(CVC)
abstract; abstract;
var public config bool bEnable; var public config bool bEnable;
var public config String SortPolicy; var public config String SortPolicy;
public static function InitConfig(int Version, int LatestVersion, E_LogLevel LogLevel) public static function InitConfig(int Version, int LatestVersion, E_LogLevel LogLevel)
{ {
`Log_TraceStatic(); `Log_TraceStatic();
switch (Version) switch (Version)
{ {
case `NO_CONFIG: case `NO_CONFIG:
ApplyDefault(LogLevel); ApplyDefault(LogLevel);
default: break; default: break;
} }
if (LatestVersion != Version) if (LatestVersion != Version)
{ {
StaticSaveConfig(); StaticSaveConfig();
} }
} }
public static function Load(E_LogLevel LogLevel) public static function Load(E_LogLevel LogLevel)
{ {
`Log_TraceStatic(); `Log_TraceStatic();
switch (Locs(default.SortPolicy)) switch (Locs(default.SortPolicy))
{ {
case "counterasc": return; case "counterasc": return;
case "counterdesc": return; case "counterdesc": return;
case "nameasc": return; case "nameasc": return;
case "namedesc": return; case "namedesc": return;
case "playtimetotalasc": return; case "playtimetotalasc": return;
case "playtimetotaldesc": return; case "playtimetotaldesc": return;
case "playtimeavgasc": return; case "playtimeavgasc": return;
case "playtimeavgdesc": return; case "playtimeavgdesc": return;
} }
`Log_Error("Can't load SortPolicy (" $ default.SortPolicy $ "), must be one of: CounterAsc CounterDesc NameAsc NameDesc PlaytimeTotalAsc PlaytimeTotalDesc PlaytimeAvgAsc PlaytimeAvgDesc"); `Log_Error("Can't load SortPolicy (" $ default.SortPolicy $ "), must be one of: CounterAsc CounterDesc NameAsc NameDesc PlaytimeTotalAsc PlaytimeTotalDesc PlaytimeAvgAsc PlaytimeAvgDesc");
default.SortPolicy = "CounterDesc"; default.SortPolicy = "CounterDesc";
} }
protected static function ApplyDefault(E_LogLevel LogLevel) protected static function ApplyDefault(E_LogLevel LogLevel)
{ {
`Log_TraceStatic(); `Log_TraceStatic();
default.bEnable = false; default.bEnable = false;
default.SortPolicy = "CounterDesc"; default.SortPolicy = "CounterDesc";
} }
defaultproperties defaultproperties
{ {
} }

View File

@ -1,70 +1,70 @@
class MapStats extends Object class MapStats extends Object
config(MapStats); config(MapStats);
struct MapStatEntry struct MapStatEntry
{ {
var String Name; // map var String Name; // map
var int Counter; // play count var int Counter; // play count
var int PlayTimeTotal; // minutes total var int PlayTimeTotal; // minutes total
var int PlayTimeAvg; // minutes per map var int PlayTimeAvg; // minutes per map
}; };
var config array<MapStatEntry> MapStat; var config array<MapStatEntry> MapStat;
static delegate int CounterAsc (MapStatEntry A, MapStatEntry B) { return B.Counter < A.Counter ? -1 : 0; } static delegate int CounterAsc (MapStatEntry A, MapStatEntry B) { return B.Counter < A.Counter ? -1 : 0; }
static delegate int CounterDesc (MapStatEntry A, MapStatEntry B) { return A.Counter < B.Counter ? -1 : 0; } static delegate int CounterDesc (MapStatEntry A, MapStatEntry B) { return A.Counter < B.Counter ? -1 : 0; }
static delegate int NameAsc (MapStatEntry A, MapStatEntry B) { return B.Name < A.Name ? -1 : 0; } static delegate int NameAsc (MapStatEntry A, MapStatEntry B) { return B.Name < A.Name ? -1 : 0; }
static delegate int NameDesc (MapStatEntry A, MapStatEntry B) { return A.Name < B.Name ? -1 : 0; } static delegate int NameDesc (MapStatEntry A, MapStatEntry B) { return A.Name < B.Name ? -1 : 0; }
static delegate int PlayTimeTotalAsc (MapStatEntry A, MapStatEntry B) { return B.PlayTimeTotal < A.PlayTimeTotal ? -1 : 0; } static delegate int PlayTimeTotalAsc (MapStatEntry A, MapStatEntry B) { return B.PlayTimeTotal < A.PlayTimeTotal ? -1 : 0; }
static delegate int PlayTimeTotalDesc (MapStatEntry A, MapStatEntry B) { return A.PlayTimeTotal < B.PlayTimeTotal ? -1 : 0; } static delegate int PlayTimeTotalDesc (MapStatEntry A, MapStatEntry B) { return A.PlayTimeTotal < B.PlayTimeTotal ? -1 : 0; }
static delegate int PlayTimeAvgAsc (MapStatEntry A, MapStatEntry B) { return B.PlayTimeAvg < A.PlayTimeAvg ? -1 : 0; } static delegate int PlayTimeAvgAsc (MapStatEntry A, MapStatEntry B) { return B.PlayTimeAvg < A.PlayTimeAvg ? -1 : 0; }
static delegate int PlayTimeAvgDesc (MapStatEntry A, MapStatEntry B) { return A.PlayTimeAvg < B.PlayTimeAvg ? -1 : 0; } static delegate int PlayTimeAvgDesc (MapStatEntry A, MapStatEntry B) { return A.PlayTimeAvg < B.PlayTimeAvg ? -1 : 0; }
static function SortMapStat(String SortPolicy, E_LogLevel LogLevel) static function SortMapStat(String SortPolicy, E_LogLevel LogLevel)
{ {
`Log_TraceStatic(); `Log_TraceStatic();
switch (Locs(SortPolicy)) switch (Locs(SortPolicy))
{ {
case "counterasc": default.MapStat.Sort(CounterAsc); break; case "counterasc": default.MapStat.Sort(CounterAsc); break;
case "counterdesc": default.MapStat.Sort(CounterDesc); break; case "counterdesc": default.MapStat.Sort(CounterDesc); break;
case "nameasc": default.MapStat.Sort(NameAsc); break; case "nameasc": default.MapStat.Sort(NameAsc); break;
case "namedesc": default.MapStat.Sort(NameDesc); break; case "namedesc": default.MapStat.Sort(NameDesc); break;
case "playtimetotalasc": default.MapStat.Sort(PlayTimeTotalAsc); break; case "playtimetotalasc": default.MapStat.Sort(PlayTimeTotalAsc); break;
case "playtimetotaldesc": default.MapStat.Sort(PlayTimeTotalDesc); break; case "playtimetotaldesc": default.MapStat.Sort(PlayTimeTotalDesc); break;
case "playtimeavgasc": default.MapStat.Sort(PlayTimeAvgAsc); break; case "playtimeavgasc": default.MapStat.Sort(PlayTimeAvgAsc); break;
case "playtimeavgdesc": default.MapStat.Sort(PlayTimeAvgDesc); break; case "playtimeavgdesc": default.MapStat.Sort(PlayTimeAvgDesc); break;
} }
} }
static function IncMapStat(String Map, int PlayTime, String SortPolicy, E_LogLevel LogLevel) static function IncMapStat(String Map, int PlayTime, String SortPolicy, E_LogLevel LogLevel)
{ {
local int MapStatEntryIndex; local int MapStatEntryIndex;
local MapStatEntry NewEntry; local MapStatEntry NewEntry;
`Log_TraceStatic(); `Log_TraceStatic();
MapStatEntryIndex = default.MapStat.Find('Name', Map); MapStatEntryIndex = default.MapStat.Find('Name', Map);
if (MapStatEntryIndex == INDEX_NONE) if (MapStatEntryIndex == INDEX_NONE)
{ {
NewEntry.Name = Map; NewEntry.Name = Map;
NewEntry.Counter = 1; NewEntry.Counter = 1;
NewEntry.PlayTimeTotal = PlayTime; NewEntry.PlayTimeTotal = PlayTime;
NewEntry.PlayTimeAvg = PlayTime; NewEntry.PlayTimeAvg = PlayTime;
default.MapStat.AddItem(NewEntry); default.MapStat.AddItem(NewEntry);
} }
else else
{ {
default.MapStat[MapStatEntryIndex].Counter++; default.MapStat[MapStatEntryIndex].Counter++;
default.MapStat[MapStatEntryIndex].PlayTimeTotal += PlayTime; default.MapStat[MapStatEntryIndex].PlayTimeTotal += PlayTime;
default.MapStat[MapStatEntryIndex].PlayTimeAvg = default.MapStat[MapStatEntryIndex].PlayTimeTotal / default.MapStat[MapStatEntryIndex].Counter; default.MapStat[MapStatEntryIndex].PlayTimeAvg = default.MapStat[MapStatEntryIndex].PlayTimeTotal / default.MapStat[MapStatEntryIndex].Counter;
} }
SortMapStat(SortPolicy, LogLevel); SortMapStat(SortPolicy, LogLevel);
StaticSaveConfig(); StaticSaveConfig();
} }
DefaultProperties defaultproperties
{ {
} }

View File

@ -1,57 +1,57 @@
class MapVote extends Object class MapVote extends Object
config(CVC) config(CVC)
abstract; abstract;
var public config String DefaultNextMap; // Any, Official, Custom, KF-<MapName> var public config String DefaultNextMap; // Any, Official, Custom, KF-<MapName>
var public config bool bRandomizeNextMap; var public config bool bRandomizeNextMap;
public static function InitConfig(int Version, int LatestVersion, E_LogLevel LogLevel) public static function InitConfig(int Version, int LatestVersion, E_LogLevel LogLevel)
{ {
`Log_TraceStatic(); `Log_TraceStatic();
switch (Version) switch (Version)
{ {
case `NO_CONFIG: case `NO_CONFIG:
ApplyDefault(LogLevel); ApplyDefault(LogLevel);
default: break; default: break;
} }
if (LatestVersion != Version) if (LatestVersion != Version)
{ {
StaticSaveConfig(); StaticSaveConfig();
} }
} }
public static function Load(E_LogLevel LogLevel) public static function Load(E_LogLevel LogLevel)
{ {
local String LowerDefaultNextMap; local String LowerDefaultNextMap;
`Log_TraceStatic(); `Log_TraceStatic();
LowerDefaultNextMap = Locs(default.DefaultNextMap); LowerDefaultNextMap = Locs(default.DefaultNextMap);
switch (LowerDefaultNextMap) switch (LowerDefaultNextMap)
{ {
case "any": return; case "any": return;
case "official": return; case "official": return;
case "custom": return; case "custom": return;
default: if (Left(LowerDefaultNextMap, 3) == "kf-") return; default: if (Left(LowerDefaultNextMap, 3) == "kf-") return;
} }
`Log_Error("Can't load DefaultNextMap (" $ default.DefaultNextMap $ "), must be one of: Any Official Custom KF-<MapName>"); `Log_Error("Can't load DefaultNextMap (" $ default.DefaultNextMap $ "), must be one of: Any Official Custom KF-<MapName>");
default.DefaultNextMap = "Any"; default.DefaultNextMap = "Any";
} }
protected static function ApplyDefault(E_LogLevel LogLevel) protected static function ApplyDefault(E_LogLevel LogLevel)
{ {
`Log_TraceStatic(); `Log_TraceStatic();
default.bRandomizeNextMap = true; default.bRandomizeNextMap = true;
default.DefaultNextMap = "Any"; default.DefaultNextMap = "Any";
} }
defaultproperties defaultproperties
{ {
} }

60
CVC/Classes/Mut.uc Normal file
View File

@ -0,0 +1,60 @@
class Mut extends KFMutator;
var private CVC CVC;
public simulated function bool SafeDestroy()
{
return (bPendingDelete || bDeleteMe || Destroy());
}
public event PreBeginPlay()
{
Super.PreBeginPlay();
if (WorldInfo.NetMode == NM_Client) return;
foreach WorldInfo.DynamicActors(class'CVC', CVC)
{
break;
}
if (CVC == None)
{
CVC = WorldInfo.Spawn(class'CVC');
}
if (CVC == None)
{
`Log_Base("FATAL: Can't Spawn 'CVC'");
SafeDestroy();
}
}
public function AddMutator(Mutator M)
{
if (M == Self) return;
if (M.Class == Class)
Mut(M).SafeDestroy();
else
Super.AddMutator(M);
}
public function NotifyLogin(Controller C)
{
Super.NotifyLogin(C);
CVC.NotifyLogin(C);
}
public function NotifyLogout(Controller C)
{
Super.NotifyLogout(C);
CVC.NotifyLogout(C);
}
defaultproperties
{
GroupNames.Add("VoteCollector")
}

View File

@ -1,19 +1,19 @@
class PauseVote extends BaseVote class PauseVote extends BaseVote
config(CVC) config(CVC)
abstract; abstract;
protected static function ApplyDefault(E_LogLevel LogLevel) protected static function ApplyDefault(E_LogLevel LogLevel)
{ {
`Log_TraceStatic(); `Log_TraceStatic();
Super.ApplyDefault(LogLevel); Super.ApplyDefault(LogLevel);
default.PositiveColorHex = class'KFLocalMessage'.default.GameColor; default.PositiveColorHex = class'KFLocalMessage'.default.GameColor;
default.bChatNotifications = false; default.bChatNotifications = false;
default.bHudNotifications = false; default.bHudNotifications = false;
} }
defaultproperties defaultproperties
{ {
} }

View File

@ -1,19 +1,19 @@
class SkipTraderVote extends BaseVote class SkipTraderVote extends BaseVote
config(CVC) config(CVC)
abstract; abstract;
protected static function ApplyDefault(E_LogLevel LogLevel) protected static function ApplyDefault(E_LogLevel LogLevel)
{ {
`Log_TraceStatic(); `Log_TraceStatic();
Super.ApplyDefault(LogLevel); Super.ApplyDefault(LogLevel);
default.PositiveColorHex = class'KFLocalMessage'.default.GameColor; default.PositiveColorHex = class'KFLocalMessage'.default.GameColor;
default.bChatNotifications = false; default.bChatNotifications = false;
default.bHudNotifications = false; default.bHudNotifications = false;
} }
defaultproperties defaultproperties
{ {
} }

View File

@ -1,54 +1,54 @@
class StartWaveKickProtection extends Object class StartWaveKickProtection extends Object
config(CVC) config(CVC)
abstract; abstract;
var public config int Waves; var public config int Waves;
var public config int MinLevel; var public config int MinLevel;
public static function InitConfig(int Version, int LatestVersion, E_LogLevel LogLevel) public static function InitConfig(int Version, int LatestVersion, E_LogLevel LogLevel)
{ {
`Log_TraceStatic(); `Log_TraceStatic();
switch (Version) switch (Version)
{ {
case `NO_CONFIG: case `NO_CONFIG:
ApplyDefault(LogLevel); ApplyDefault(LogLevel);
default: break; default: break;
} }
if (LatestVersion != Version) if (LatestVersion != Version)
{ {
StaticSaveConfig(); StaticSaveConfig();
} }
} }
public static function Load(E_LogLevel LogLevel) public static function Load(E_LogLevel LogLevel)
{ {
`Log_TraceStatic(); `Log_TraceStatic();
if (default.Waves < 0) if (default.Waves < 0)
{ {
`Log_Error("Waves" @ "(" $ default.Waves $ ")" @ "must be greater than or equal 0"); `Log_Error("Waves" @ "(" $ default.Waves $ ")" @ "must be greater than or equal 0");
default.Waves = 0; default.Waves = 0;
} }
if (default.MinLevel < 0 || default.MinLevel > 25) if (default.MinLevel < 0 || default.MinLevel > 25)
{ {
`Log_Error("MinLevel" @ "(" $ default.MinLevel $ ")" @ "must be in range 0-25"); `Log_Error("MinLevel" @ "(" $ default.MinLevel $ ")" @ "must be in range 0-25");
default.MinLevel = 0; default.MinLevel = 0;
} }
} }
protected static function ApplyDefault(E_LogLevel LogLevel) protected static function ApplyDefault(E_LogLevel LogLevel)
{ {
`Log_TraceStatic(); `Log_TraceStatic();
default.Waves = 0; default.Waves = 0;
default.MinLevel = 0; default.MinLevel = 0;
} }
defaultproperties defaultproperties
{ {
} }

View File

@ -1,21 +1,21 @@
class _Logger extends Object class _Logger extends Object
abstract; abstract;
enum E_LogLevel enum E_LogLevel
{ {
LL_WrongLevel, LL_WrongLevel,
LL_None, LL_None,
LL_Fatal, LL_Fatal,
LL_Error, LL_Error,
LL_Warning, LL_Warning,
LL_Kick, LL_Kick,
LL_Info, LL_Info,
LL_Debug, LL_Debug,
LL_Trace, LL_Trace,
LL_All LL_All
}; };
defaultproperties defaultproperties
{ {
} }

View File

@ -1,2 +1,2 @@
// Constants // Constants
`define NO_CONFIG 0 `define NO_CONFIG 0

View File

@ -1,3 +1,3 @@
// Imports // Imports
`include(Logger.uci) `include(Logger.uci)
`include(Constants.uci) `include(Constants.uci)

View File

@ -1,16 +1,16 @@
// Logger // Logger
`define Log_Tag 'CVC' `define Log_Tag 'CVC'
`define LocationStatic "`{ClassName}::" $ GetFuncName() `define LocationStatic "`{ClassName}::" $ GetFuncName()
`define Log_Base(msg, cond) `log(`msg `if(`cond), `cond`{endif}, `Log_Tag) `define Log_Base(msg, cond) `log(`msg `if(`cond), `cond`{endif}, `Log_Tag)
`define Log_Fatal(msg) `log("FATAL:" @ `msg, (LogLevel >= LL_Fatal), `Log_Tag) `define Log_Fatal(msg) `log("FATAL:" @ `msg, (LogLevel >= LL_Fatal), `Log_Tag)
`define Log_Error(msg) `log("ERROR:" @ `msg, (LogLevel >= LL_Error), `Log_Tag) `define Log_Error(msg) `log("ERROR:" @ `msg, (LogLevel >= LL_Error), `Log_Tag)
`define Log_Warn(msg) `log("WARN:" @ `msg, (LogLevel >= LL_Warning), `Log_Tag) `define Log_Warn(msg) `log("WARN:" @ `msg, (LogLevel >= LL_Warning), `Log_Tag)
`define Log_Kick(msg) `log("KICK:" @ `msg, (LogLevel >= LL_Kick), `Log_Tag) `define Log_Kick(msg) `log("KICK:" @ `msg, (LogLevel >= LL_Kick), `Log_Tag)
`define Log_Info(msg) `log("INFO:" @ `msg, (LogLevel >= LL_Info), `Log_Tag) `define Log_Info(msg) `log("INFO:" @ `msg, (LogLevel >= LL_Info), `Log_Tag)
`define Log_Debug(msg) `log("DEBUG:" @ `msg, (LogLevel >= LL_Debug), `Log_Tag) `define Log_Debug(msg) `log("DEBUG:" @ `msg, (LogLevel >= LL_Debug), `Log_Tag)
`define Log_Trace(msg) `log("TRACE:" @ `Location `if(`msg) @ `msg`{endif}, (LogLevel >= LL_Trace), `Log_Tag) `define Log_Trace(msg) `log("TRACE:" @ `Location `if(`msg) @ `msg`{endif}, (LogLevel >= LL_Trace), `Log_Tag)
`define Log_TraceStatic(msg) `log("TRACE:" @ `LocationStatic `if(`msg) @ `msg`{endif}, (LogLevel >= LL_Trace), `Log_Tag) `define Log_TraceStatic(msg) `log("TRACE:" @ `LocationStatic `if(`msg) @ `msg`{endif}, (LogLevel >= LL_Trace), `Log_Tag)

BIN
Localization/CHN/CVC.chn Normal file

Binary file not shown.

BIN
Localization/CHT/CVC.cht Normal file

Binary file not shown.

Binary file not shown.

Binary file not shown.

View File

@ -1,4 +1,4 @@
[img]https://img.shields.io/static/v1?logo=GitHub&labelColor=gray&color=blue&logoColor=white&label=&message=Open Source[/img] [img]https://img.shields.io/github/license/GenZmeY/KF2-ControlledVoteCollector[/img] [img]https://img.shields.io/steam/subscriptions/2847465899[/img] [img]https://img.shields.io/steam/favorites/2847465899[/img] [img]https://img.shields.io/steam/update-date/2847465899[/img] [url=https://steamcommunity.com/sharedfiles/filedetails/changelog/2847465899][img]https://img.shields.io/github/v/tag/GenZmeY/KF2-ControlledVoteCollector[/img][/url] [img]https://img.shields.io/static/v1?logo=GitHub&labelColor=gray&color=blue&logoColor=white&label=&message=Open Source[/img] [img]https://img.shields.io/github/license/GenZmeY/KF2-ControlledVoteCollector[/img] [img]https://img.shields.io/steam/favorites/2847465899[/img] [img]https://img.shields.io/steam/update-date/2847465899[/img] [url=https://steamcommunity.com/sharedfiles/filedetails/changelog/2847465899][img]https://img.shields.io/github/v/tag/GenZmeY/KF2-ControlledVoteCollector[/img][/url]
[h1]Description[/h1] [h1]Description[/h1]
New vote collector with improvements and features. New vote collector with improvements and features.
@ -16,7 +16,10 @@ New vote collector with improvements and features.
[i](it would be logical to separate these features into several mutators, but this is a bad idea for technical reasons)[/i] [i](it would be logical to separate these features into several mutators, but this is a bad idea for technical reasons)[/i]
[h1]Whitelisted?[/h1] [h1]Whitelisted?[/h1]
No. This mod is not whitelisted and will de-rank your server. Any XP gained will not be saved. [b]No.[/b] But I really hope that it will be whitelisted.
[b]⚠️ I submitted whitelist request here:[/b]
https://forums.tripwireinteractive.com/index.php?threads/whitelisting-mods-and-mutators.120340/post-2353667
[h1]Usage (server)[/h1] [h1]Usage (server)[/h1]
[b]Note:[/b] [i]If you don't understand what is written here, read the article [url=https://wiki.killingfloor2.com/index.php?title=Dedicated_Server_(Killing_Floor_2)][u]Dedicated Server (KF2 wiki)[/u][/url] before following these instructions.[/i] [b]Note:[/b] [i]If you don't understand what is written here, read the article [url=https://wiki.killingfloor2.com/index.php?title=Dedicated_Server_(Killing_Floor_2)][u]Dedicated Server (KF2 wiki)[/u][/url] before following these instructions.[/i]
@ -28,11 +31,11 @@ No. This mod is not whitelisted and will de-rank your server. Any XP gained will
[*]Add the following string to the [b][OnlineSubsystemSteamworks.KFWorkshopSteamworks][/b] section (create one if it doesn't exist): [*]Add the following string to the [b][OnlineSubsystemSteamworks.KFWorkshopSteamworks][/b] section (create one if it doesn't exist):
[b]ServerSubscribedWorkshopItems=2847465899[/b] [b]ServerSubscribedWorkshopItems=2847465899[/b]
[*]Start the server and wait until the mutator is downloading; [*]Start the server and wait until the mutator is downloading;
[*]Add mutator to server start parameters: [b]?Mutator=CVC.CVCMut[/b] and restart the server. [*]Add mutator to server start parameters: [b]?Mutator=CVC.Mut[/b] and restart the server.
[/olist] [/olist]
[h1]Setup (KFCVC.ini)[/h1] [h1]Setup (KFCVC.ini)[/h1]
Config will be created at the first start. Config will be created at the first start[b]*[/b].
[b][CVC.MapStat][/b] [b][CVC.MapStat][/b]
[list] [list]
@ -69,7 +72,8 @@ This section sets the next map when no one voted for the map.
[*][b]PositiveColorHex[/b] - hex color for yes vote in chat; [*][b]PositiveColorHex[/b] - hex color for yes vote in chat;
[*][b]NegativeColorHex[/b] - hex color for no vote in chat; [*][b]NegativeColorHex[/b] - hex color for no vote in chat;
[*][b]bHudNotifications[/b] - set to [b]True[/b] to see player votes in HUD; [*][b]bHudNotifications[/b] - set to [b]True[/b] to see player votes in HUD;
[*][b]DefferedClearHUD[/b] - HUD notification will remain on the screen for the specified number of seconds after voting ends. [*][b]DefferedClearHUD[/b] - HUD notification will remain on the screen for the specified number of seconds after voting ends;
[*][b]VoteTime[/b] - time in seconds for voting (will be automatically reduced if it exceeds the trader's remaining time).
[/list] [/list]
[b][CVC.PauseVote][/b] [b][CVC.PauseVote][/b]
@ -78,7 +82,8 @@ This section sets the next map when no one voted for the map.
[*][b]PositiveColorHex[/b] - hex color for yes vote in chat; [*][b]PositiveColorHex[/b] - hex color for yes vote in chat;
[*][b]NegativeColorHex[/b] - hex color for no vote in chat; [*][b]NegativeColorHex[/b] - hex color for no vote in chat;
[*][b]bHudNotifications[/b] - set to [b]True[/b] to see player votes in HUD; [*][b]bHudNotifications[/b] - set to [b]True[/b] to see player votes in HUD;
[*][b]DefferedClearHUD[/b] - HUD notification will remain on the screen for the specified number of seconds after voting ends. [*][b]DefferedClearHUD[/b] - HUD notification will remain on the screen for the specified number of seconds after voting ends;
[*][b]VoteTime[/b] - time in seconds for voting (will be automatically reduced if it exceeds the trader's remaining time).
[/list] [/list]
[b][CVC.KickVote][/b] [b][CVC.KickVote][/b]
@ -92,7 +97,8 @@ This section sets the next map when no one voted for the map.
[*][b]DefferedClearHUD[/b] - HUD notification will remain on the screen for the specified number of seconds after voting ends. [*][b]DefferedClearHUD[/b] - HUD notification will remain on the screen for the specified number of seconds after voting ends.
[*][b]bLogKickVote[/b] - set to [b]True[/b] to log information about every kick vote; [*][b]bLogKickVote[/b] - set to [b]True[/b] to log information about every kick vote;
[*][b]MinVotingPlayersToStartKickVote[/b] - minimum number of voting players to start kick voting; [*][b]MinVotingPlayersToStartKickVote[/b] - minimum number of voting players to start kick voting;
[*][b]MaxKicks[/b] - maximum number of kicks per game. [*][b]MaxKicks[/b] - maximum number of kicks per game;
[*][b]VoteTime[/b] - time in seconds for voting.
[/list] [/list]
[b][CVC.KickProtected][/b] [b][CVC.KickProtected][/b]
@ -104,7 +110,7 @@ This section sets the next map when no one voted for the map.
[b][CVC.StartWaveKickProtection][/b] [b][CVC.StartWaveKickProtection][/b]
In this section, the system for preventing early kicks is configured (especially for lazy ass admins like me who don't want to consider player complaints about this). In this section, the system for preventing early kicks is configured (especially for lazy ass admins like me who don't want to consider player complaints about this).
[list] [list]
[*][b]Waves[/b] - the number of waves during which a new player has kick protection and cannot vote for a kick; [*][b]Waves[/b] - the number of waves during which a new player has kick protection and cannot start kick vote;
[*][b]MinLevel[/b] - the minimum level that a player needs to have in order to receive protection from a kick after joining the server. [*][b]MinLevel[/b] - the minimum level that a player needs to have in order to receive protection from a kick after joining the server.
[/list] [/list]
@ -117,5 +123,16 @@ When the player has played the specified number of [b]Waves[/b], he loses the ki
The [b]MinLevel[/b] parameter specifies an exception to these rules, giving kick protection only to players above or equal the specified level. All players can vote to exclude players with an unsuitable level, regardless of whether they have played enough [b]Waves[/b] or not. This allows to remove low-level players without waiting for them to screw up in the game. The [b]MinLevel[/b] parameter specifies an exception to these rules, giving kick protection only to players above or equal the specified level. All players can vote to exclude players with an unsuitable level, regardless of whether they have played enough [b]Waves[/b] or not. This allows to remove low-level players without waiting for them to screw up in the game.
[h1]Troubleshooting[/h1]
[b](*)[/b] If your config is not created for some reason, create it manually with the following content:
[b][CVC.CVC]
Version=0
[/b]
Then start the server and check the file again - config content should be generated.
[h1]Translators:[/h1]
[url=https://steamcommunity.com/profiles/76561199126205919]cheungfatzong[/url] - Traditional [CHT] and Simplified [CHN] Chinese.
[h1]Sources[/h1] [h1]Sources[/h1]
[url=https://github.com/GenZmeY/KF2-ControlledVoteCollector]https://github.com/GenZmeY/KF2-ControlledVoteCollector[/url] [b](GNU GPLv3)[/b] [url=https://github.com/GenZmeY/KF2-ControlledVoteCollector]https://github.com/GenZmeY/KF2-ControlledVoteCollector[/url] [b](GNU GPLv3)[/b]

View File

@ -1,16 +1,16 @@
# Controlled Vote Collector # Controlled Vote Collector
[![Steam Workshop](https://img.shields.io/static/v1?message=workshop&logo=steam&labelColor=gray&color=blue&logoColor=white&label=steam%20)](https://steamcommunity.com/sharedfiles/filedetails/?id=2847465899) [![Steam Workshop](https://img.shields.io/static/v1?message=workshop&logo=steam&labelColor=gray&color=blue&logoColor=white&label=steam%20)](https://steamcommunity.com/sharedfiles/filedetails/?id=2847465899)
[![Steam Subscriptions](https://img.shields.io/steam/subscriptions/2847465899)](https://steamcommunity.com/sharedfiles/filedetails/?id=2847465899) [![Steam Downloads](https://img.shields.io/steam/downloads/2847465899)](https://steamcommunity.com/sharedfiles/filedetails/?id=2848836389)
[![Steam Favorites](https://img.shields.io/steam/favorites/2847465899)](https://steamcommunity.com/sharedfiles/filedetails/?id=2847465899) [![Steam Favorites](https://img.shields.io/steam/favorites/2847465899)](https://steamcommunity.com/sharedfiles/filedetails/?id=2847465899)
[![Steam Update Date](https://img.shields.io/steam/update-date/2847465899)](https://steamcommunity.com/sharedfiles/filedetails/?id=2847465899) [![MegaLinter](https://github.com/GenZmeY/KF2-ControlledVoteCollector/actions/workflows/mega-linter.yml/badge.svg?branch=master)](https://github.com/GenZmeY/KF2-ControlledVoteCollector/actions/workflows/mega-linter.yml)
[![GitHub tag (latest by date)](https://img.shields.io/github/v/tag/GenZmeY/KF2-ControlledVoteCollector)](https://github.com/GenZmeY/KF2-ControlledVoteCollector/tags) [![GitHub tag (latest by date)](https://img.shields.io/github/v/tag/GenZmeY/KF2-ControlledVoteCollector)](https://github.com/GenZmeY/KF2-ControlledVoteCollector/tags)
[![GitHub](https://img.shields.io/github/license/GenZmeY/KF2-ControlledVoteCollector)](LICENSE) [![GitHub](https://img.shields.io/github/license/GenZmeY/KF2-ControlledVoteCollector)](LICENSE)
# Description ## Description
New vote collector with improvements and features. New vote collector with improvements and features.
# Features ## Features
- map statistics; - map statistics;
- default/next map setting; - default/next map setting;
- anonymous or public voting; - anonymous or public voting;
@ -18,10 +18,10 @@ New vote collector with improvements and features.
- kick voting setup; - kick voting setup;
- early kick protection. - early kick protection.
# Usage & Setup ## Usage & Setup
[See steam workshop page](https://steamcommunity.com/sharedfiles/filedetails/?id=2847465899) [See steam workshop page](https://steamcommunity.com/sharedfiles/filedetails/?id=2847465899)
# Build ## Build
**Note:** If you want to build/test/brew/publish a mutator without git-bash and/or scripts, follow [these instructions](https://tripwireinteractive.atlassian.net/wiki/spaces/KF2SW/pages/26247172/KF2+Code+Modding+How-to) instead of what is described here. **Note:** If you want to build/test/brew/publish a mutator without git-bash and/or scripts, follow [these instructions](https://tripwireinteractive.atlassian.net/wiki/spaces/KF2SW/pages/26247172/KF2+Code+Modding+How-to) instead of what is described here.
1. Install [Killing Floor 2](https://store.steampowered.com/app/232090/Killing_Floor_2/), Killing Floor 2 - SDK and [git for windows](https://git-scm.com/download/win); 1. Install [Killing Floor 2](https://store.steampowered.com/app/232090/Killing_Floor_2/), Killing Floor 2 - SDK and [git for windows](https://git-scm.com/download/win);
2. open git-bash and go to any folder where you want to store sources: 2. open git-bash and go to any folder where you want to store sources:
@ -35,9 +35,9 @@ New vote collector with improvements and features.
5. The compiled files will be here: 5. The compiled files will be here:
`C:\Users\<USERNAME>\Documents\My Games\KillingFloor2\KFGame\Unpublished\BrewedPC\Script\` `C:\Users\<USERNAME>\Documents\My Games\KillingFloor2\KFGame\Unpublished\BrewedPC\Script\`
# Bug reports ## Bug reports
If you find a bug, go to the [issue page](https://github.com/GenZmeY/KF2-ControlledVoteCollector/issues) and check if there is a description of your bug. If not, create a new issue. If you find a bug, go to the [issue page](https://github.com/GenZmeY/KF2-ControlledVoteCollector/issues) and check if there is a description of your bug. If not, create a new issue.
Describe what the bug looks like and how reproduce it. Describe what the bug looks like and how reproduce it.
# License ## License
[GNU GPLv3](LICENSE) [![license](https://www.gnu.org/graphics/gplv3-with-text-136x68.png)](LICENSE)

View File

@ -7,15 +7,24 @@ StripSource="True"
# Mutators to be compiled # Mutators to be compiled
# Specify them with a space as a separator, # Specify them with a space as a separator,
# Mutators will be compiled in the specified order # Mutators will be compiled in the specified order
PackageBuildOrder="CVC" PackageBuildOrder="CVC"
### Brew parameters ###
# Packages you want to brew using @peelz's patched KFEditor.
# Useful for cases where regular brew doesn't put *.upk inside the package.
# Specify them with a space as a separator,
# The order doesn't matter
PackagePeelzBrew=""
### Steam Workshop upload parameters ### ### Steam Workshop upload parameters ###
# Mutators that will be uploaded to the workshop # Mutators that will be uploaded to the workshop
# Specify them with a space as a separator, # Specify them with a space as a separator,
# The order doesn't matter # The order doesn't matter
PackageUpload="CVC" PackageUpload="CVC"

2
tools

@ -1 +1 @@
Subproject commit 2f173aad7a6f4578574764801136a0d86e830653 Subproject commit fb458ac61f7e6c6426b8dff366dd5e7499e0d95f