Compare commits
38 Commits
Author | SHA1 | Date | |
---|---|---|---|
522b609ebc | |||
8223dec827 | |||
e680270722 | |||
1f6de98e2f | |||
469c8bbd96 | |||
48e1b9ff1d | |||
464ba0f492 | |||
afadeba04c | |||
1c1b7f6e90 | |||
690aa2ba6f | |||
158c6b8a2e | |||
e354b714ca | |||
6fb5902567 | |||
8a26a40e90 | |||
edb2334ac8 | |||
c37b64c3be | |||
9635fb5007 | |||
e595daf7ca | |||
9b47405939 | |||
345e765a0e | |||
c724de802b | |||
da1f19cf9d | |||
777b2389c1 | |||
66d7dce3d0 | |||
b66f2e9fef | |||
af861b9888 | |||
5f4617e25f | |||
5dd4279dd4 | |||
31db3849be | |||
cdacf03d40 | |||
8dfe63771b | |||
97866c1a55 | |||
2461bda5f7 | |||
8d02c1e0a3 | |||
3379596de4 | |||
a881c1ef93 | |||
5faba66cb1 | |||
51aba2c1a1 |
33
.editorconfig
Normal file
33
.editorconfig
Normal 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
115
.github/workflows/mega-linter.yml
vendored
Normal 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
2
.gitignore
vendored
Normal file
@ -0,0 +1,2 @@
|
||||
*.psd
|
||||
/ignore
|
@ -7,6 +7,7 @@ var public config String NegativeColorHex;
|
||||
var public config bool bChatNotifications;
|
||||
var public config bool bHudNotifications;
|
||||
var public config float DefferedClearHUD;
|
||||
var public config int VoteTime;
|
||||
|
||||
public static function InitConfig(int Version, int LatestVersion, E_LogLevel LogLevel)
|
||||
{
|
||||
@ -17,6 +18,9 @@ public static function InitConfig(int Version, int LatestVersion, E_LogLevel Log
|
||||
case `NO_CONFIG:
|
||||
ApplyDefault(LogLevel);
|
||||
|
||||
case 1:
|
||||
default.VoteTime = class'KFVoteCollector'.default.VoteTime;
|
||||
|
||||
default: break;
|
||||
}
|
||||
|
||||
@ -47,6 +51,12 @@ public static function Load(E_LogLevel LogLevel)
|
||||
`Log_Error("DefferedClearHUD" @ "(" $ default.DefferedClearHUD $ ")" @ "must be greater than 0");
|
||||
default.DefferedClearHUD = 0.0f;
|
||||
}
|
||||
|
||||
if (default.VoteTime <= 0 || default.VoteTime > 255)
|
||||
{
|
||||
`Log_Error("VoteTime" @ "(" $ default.VoteTime $ ")" @ "must be in range 1-255");
|
||||
default.VoteTime = class'KFVoteCollector'.default.VoteTime;
|
||||
}
|
||||
}
|
||||
|
||||
protected static function ApplyDefault(E_LogLevel LogLevel)
|
||||
@ -58,6 +68,7 @@ protected static function ApplyDefault(E_LogLevel LogLevel)
|
||||
default.PositiveColorHex = class'KFLocalMessage'.default.EventColor;
|
||||
default.NegativeColorHex = class'KFLocalMessage'.default.InteractionColor;
|
||||
default.DefferedClearHUD = 1.0f;
|
||||
default.VoteTime = class'KFVoteCollector'.default.VoteTime;
|
||||
}
|
||||
|
||||
protected static function bool IsValidHexColor(String HexColor, E_LogLevel LogLevel)
|
||||
|
@ -2,7 +2,7 @@ class CVC extends Info
|
||||
dependson(CVC_LocalMessage)
|
||||
config(CVC);
|
||||
|
||||
const LatestVersion = 1;
|
||||
const LatestVersion = 2;
|
||||
|
||||
const CfgKickProtected = class'KickProtected';
|
||||
const CfgKickVote = class'KickVote';
|
||||
@ -79,6 +79,8 @@ private function PreInit()
|
||||
case `NO_CONFIG:
|
||||
`Log_Info("Config created");
|
||||
|
||||
case 1:
|
||||
|
||||
case MaxInt:
|
||||
`Log_Info("Config updated to version"@LatestVersion);
|
||||
break;
|
||||
@ -181,6 +183,13 @@ public function bool PlayerIsKickProtected(PlayerReplicationInfo PRI)
|
||||
return (KickProtectedPlayers.Find('Uid', PRI.UniqueId.Uid) != INDEX_NONE);
|
||||
}
|
||||
|
||||
public function bool PlayerPerkLoaded(PlayerReplicationInfo PRI)
|
||||
{
|
||||
`Log_Trace();
|
||||
|
||||
return (KFPlayerReplicationInfo(PRI) != None && KFPlayerReplicationInfo(PRI).CurrentPerkClass != None);
|
||||
}
|
||||
|
||||
public function bool PlayerIsStartWaveKickProtected(KFPlayerController KFPC)
|
||||
{
|
||||
`Log_Trace();
|
||||
@ -224,15 +233,10 @@ private function bool PlayerHasRequiredLevel(KFPlayerController KFPC)
|
||||
return (KFPRI.GetActivePerkLevel() >= CfgStartWaveKickProtection.default.MinLevel);
|
||||
}
|
||||
|
||||
public function bool PlayerCanKickVote(KFPlayerController KFPC, optional KFPlayerController KFPC_Kickee)
|
||||
public function bool PlayerCanStartKickVote(KFPlayerController KFPC, KFPlayerController KFPC_Kickee)
|
||||
{
|
||||
`Log_Trace();
|
||||
|
||||
if (KFPC_Kickee == None)
|
||||
{
|
||||
KFPC_Kickee = KFPlayerController(KFGRI.VoteCollector.CurrentKickVote.PlayerPRI.Owner);
|
||||
}
|
||||
|
||||
if (KFPC_Kickee != None)
|
||||
{
|
||||
if (KFPC == KFPC_Kickee)
|
||||
@ -329,14 +333,20 @@ public function NotifyLogin(Controller C)
|
||||
{
|
||||
`Log_Trace();
|
||||
|
||||
CreateRepInfo(C);
|
||||
if (!CreateRepInfo(C))
|
||||
{
|
||||
`Log_Error("Can't create RepInfo for:" @ C);
|
||||
}
|
||||
}
|
||||
|
||||
public function NotifyLogout(Controller C)
|
||||
{
|
||||
`Log_Trace();
|
||||
|
||||
DestroyRepInfo(C);
|
||||
if (!DestroyRepInfo(C))
|
||||
{
|
||||
`Log_Error("Can't destroy RepInfo of:" @ C);
|
||||
}
|
||||
}
|
||||
|
||||
public function bool CreateRepInfo(Controller C)
|
||||
@ -372,8 +382,8 @@ public function bool DestroyRepInfo(Controller C)
|
||||
{
|
||||
if (RepInfo.Owner == C)
|
||||
{
|
||||
RepInfo.SafeDestroy();
|
||||
RepInfos.RemoveItem(RepInfo);
|
||||
RepInfo.SafeDestroy();
|
||||
return true;
|
||||
}
|
||||
}
|
||||
@ -381,7 +391,7 @@ public function bool DestroyRepInfo(Controller C)
|
||||
return false;
|
||||
}
|
||||
|
||||
DefaultProperties
|
||||
defaultproperties
|
||||
{
|
||||
|
||||
}
|
@ -1,62 +1 @@
|
||||
class CVCMut 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)
|
||||
{
|
||||
`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
|
||||
{
|
||||
|
||||
}
|
||||
class CVCMut extends Mut; // backward compatibility
|
||||
|
@ -7,8 +7,11 @@ var private localized String PlayerIsKickProtected;
|
||||
var const String PlayerIsStartWaveKickProtectedDefault;
|
||||
var private localized String PlayerIsStartWaveKickProtected;
|
||||
|
||||
var const String PlayerCantVoteDefault;
|
||||
var private localized String PlayerCantVote;
|
||||
var const String PlayerCantStartKickVoteDefault;
|
||||
var private localized String PlayerCantStartKickVote;
|
||||
|
||||
var const String PlayerPerkIsNotLoadedDefault;
|
||||
var private localized String PlayerPerkIsNotLoaded;
|
||||
|
||||
var const String KickVoteNotEnoughPlayersDefault;
|
||||
var private localized String KickVoteNotEnoughPlayers;
|
||||
@ -28,23 +31,12 @@ var private localized String VotedPlayers;
|
||||
var const String DidntVotePlayersDefault;
|
||||
var private localized String DidntVotePlayers;
|
||||
|
||||
// TODO:
|
||||
/*
|
||||
Kick vote hud:
|
||||
start vote + only yes votes:
|
||||
header: <player vote for player>
|
||||
second line: yes votes
|
||||
|
||||
pause and skip:
|
||||
first line: voted players
|
||||
second line: dont voted players
|
||||
*/
|
||||
|
||||
enum E_CVC_LocalMessageType
|
||||
{
|
||||
CVC_PlayerIsKickProtected,
|
||||
CVC_PlayerIsStartWaveKickProtected,
|
||||
CVC_PlayerCantVote,
|
||||
CVC_PlayerCantStartKickVote,
|
||||
CVC_PlayerPerkIsNotLoaded,
|
||||
|
||||
CVC_KickVoteNotEnoughPlayers,
|
||||
CVC_KickVoteStarted,
|
||||
@ -97,8 +89,11 @@ public static function String GetLocalizedString(
|
||||
case CVC_PlayerIsStartWaveKickProtected:
|
||||
return ReplWaves(ReplKickee(default.PlayerIsStartWaveKickProtected != "" ? default.PlayerIsStartWaveKickProtected : default.PlayerIsStartWaveKickProtectedDefault, String1), String2);
|
||||
|
||||
case CVC_PlayerCantVote:
|
||||
return ReplWaves(default.PlayerCantVote != "" ? default.PlayerCantVote : default.PlayerCantVoteDefault, String1);
|
||||
case CVC_PlayerCantStartKickVote:
|
||||
return ReplWaves(default.PlayerCantStartKickVote != "" ? default.PlayerCantStartKickVote : default.PlayerCantStartKickVoteDefault, String1);
|
||||
|
||||
case CVC_PlayerPerkIsNotLoaded:
|
||||
return ReplKickee(default.PlayerPerkIsNotLoaded != "" ? default.PlayerPerkIsNotLoaded : default.PlayerPerkIsNotLoadedDefault, String1);
|
||||
|
||||
case CVC_KickVoteNotEnoughPlayers:
|
||||
return ReplWaves(default.KickVoteNotEnoughPlayers != "" ? default.KickVoteNotEnoughPlayers : default.KickVoteNotEnoughPlayersDefault, String1);
|
||||
@ -137,7 +132,7 @@ public static function String GetLocalizedString(
|
||||
return (String1 $ ":" @ class'KFCommon_LocalizedStrings'.default.NoString);
|
||||
|
||||
case CVC_VoteProgressHUD:
|
||||
return (default.VotedPlayers != "" ? default.VotedPlayers : default.VotedPlayersDefault) @ String1 $ (String2 != "" ? ("\n" $ (default.DidntVotePlayers != "" ? default.DidntVotePlayers : default.DidntVotePlayersDefault) @ String2) : "");
|
||||
return (default.DidntVotePlayers != "" ? default.DidntVotePlayers : default.DidntVotePlayersDefault) @ String2 $ (String1 != "" ? ("\n" $ (default.VotedPlayers != "" ? default.VotedPlayers : default.VotedPlayersDefault) @ String1) : "");
|
||||
}
|
||||
|
||||
return "";
|
||||
@ -147,7 +142,8 @@ defaultproperties
|
||||
{
|
||||
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)"
|
||||
PlayerCantVoteDefault = "You can't vote for kick now. You can vote when you play 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)"
|
||||
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)"
|
||||
KickVoteStartedDefault = "<kicker> has started a vote to kick <kickee>"
|
||||
KickVoteStartedForPlayerDefault = "<kicker> started voting to kick you"
|
||||
|
@ -38,6 +38,7 @@ public reliable client function WriteToChat(String Message, optional String HexC
|
||||
|
||||
if (KFPC.MyGFxManager.PartyWidget != None && KFPC.MyGFxManager.PartyWidget.PartyChatWidget != None)
|
||||
{
|
||||
KFPC.MyGFxManager.PartyWidget.PartyChatWidget.SetVisible(true);
|
||||
KFPC.MyGFxManager.PartyWidget.PartyChatWidget.AddChatMessage(Message, HexColor);
|
||||
}
|
||||
|
||||
|
@ -26,6 +26,8 @@ var private Array<S_KickVote> KickVotes;
|
||||
var public CVC CVC;
|
||||
var public E_LogLevel LogLevel;
|
||||
|
||||
var private KFGameInfo KFGI;
|
||||
|
||||
var private KFPlayerController KFPC_Kicker;
|
||||
var private KFPlayerController KFPC_Kickee;
|
||||
|
||||
@ -35,6 +37,7 @@ var private String KickeeName;
|
||||
var private String YesVotesPlayers, NoVotesPlayers;
|
||||
|
||||
var private bool AllowHudNotification;
|
||||
var private bool AllowSTPNotification; // SkipTrader and Pause
|
||||
|
||||
replication
|
||||
{
|
||||
@ -42,16 +45,26 @@ replication
|
||||
LogLevel;
|
||||
}
|
||||
|
||||
private function KFGameInfo GetKFGI()
|
||||
{
|
||||
`Log_Trace();
|
||||
|
||||
if (KFGI != None) return KFGI;
|
||||
|
||||
KFGI = KFGameInfo(WorldInfo.Game);
|
||||
|
||||
return KFGI;
|
||||
}
|
||||
|
||||
public function ServerStartVoteKick(PlayerReplicationInfo PRI_Kickee, PlayerReplicationInfo PRI_Kicker)
|
||||
{
|
||||
local Array<KFPlayerReplicationInfo> KFPRIs;
|
||||
local KFPlayerReplicationInfo KFPRI;
|
||||
local KFPlayerController KFPC;
|
||||
local KFGameInfo KFGI;
|
||||
|
||||
`Log_Trace();
|
||||
|
||||
KFGI = KFGameInfo(WorldInfo.Game);
|
||||
if (GetKFGI() == None) return;
|
||||
|
||||
KFPC_Kicker = KFPlayerController(PRI_Kicker.Owner);
|
||||
KFPC_Kickee = KFPlayerController(PRI_Kickee.Owner);
|
||||
|
||||
@ -70,16 +83,26 @@ public function ServerStartVoteKick(PlayerReplicationInfo PRI_Kickee, PlayerRepl
|
||||
return;
|
||||
}
|
||||
|
||||
if (!CVC.PlayerCanKickVote(KFPC_Kicker, KFPC_Kickee))
|
||||
if (!CVC.PlayerCanStartKickVote(KFPC_Kicker, KFPC_Kickee))
|
||||
{
|
||||
CVC.WriteToChatLocalized(
|
||||
KFPC_Kicker,
|
||||
CVC_PlayerCantVote,
|
||||
CVC_PlayerCantStartKickVote,
|
||||
CfgKickVote.default.WarningColorHex,
|
||||
String(CfgStartWaveKickProtection.default.Waves));
|
||||
return;
|
||||
}
|
||||
|
||||
if (!CVC.PlayerPerkLoaded(PRI_Kickee))
|
||||
{
|
||||
CVC.WriteToChatLocalized(
|
||||
KFPC_Kicker,
|
||||
CVC_PlayerPerkIsNotLoaded,
|
||||
CfgKickVote.default.WarningColorHex,
|
||||
KickeeName);
|
||||
return;
|
||||
}
|
||||
|
||||
if (CVC.PlayerIsStartWaveKickProtected(KFPC_Kickee))
|
||||
{
|
||||
CVC.WriteToChatLocalized(
|
||||
@ -164,7 +187,7 @@ public function ServerStartVoteKick(PlayerReplicationInfo PRI_Kickee, PlayerRepl
|
||||
GetKFPRIArray(KFPRIs);
|
||||
foreach KFPRIs(KFPRI)
|
||||
{
|
||||
KFPRI.ShowKickVote(PRI_Kickee, VoteTime, !(KFPRI == PRI_Kicker || KFPRI == PRI_Kickee || !CVC.PlayerCanKickVote(KFPlayerController(KFPRI.Owner))));
|
||||
KFPRI.ShowKickVote(PRI_Kickee, CfgKickVote.default.VoteTime, !(KFPRI == PRI_Kicker || KFPRI == PRI_Kickee));
|
||||
}
|
||||
|
||||
if (CfgKickVote.default.bChatNotifications)
|
||||
@ -180,25 +203,7 @@ public function ServerStartVoteKick(PlayerReplicationInfo PRI_Kickee, PlayerRepl
|
||||
KFPC_Kickee,
|
||||
CVC_KickVoteStartedForPlayer,
|
||||
CfgKickVote.default.NegativeColorHex,
|
||||
KickerName,
|
||||
KickeeName);
|
||||
|
||||
foreach KFPRIs(KFPRI)
|
||||
{
|
||||
if (KFPRI == PRI_Kickee)
|
||||
{
|
||||
continue;
|
||||
}
|
||||
KFPC = KFPlayerController(KFPRI.Owner);
|
||||
if (!CVC.PlayerCanKickVote(KFPC))
|
||||
{
|
||||
CVC.WriteToChatLocalized(
|
||||
KFPC,
|
||||
CVC_PlayerCantVote,
|
||||
CfgKickVote.default.WarningColorHex,
|
||||
String(CfgStartWaveKickProtection.default.Waves));
|
||||
}
|
||||
}
|
||||
KickerName);
|
||||
}
|
||||
else
|
||||
{
|
||||
@ -216,7 +221,7 @@ public function ServerStartVoteKick(PlayerReplicationInfo PRI_Kickee, PlayerRepl
|
||||
AllowHudNotification = bTraderIsOpen;
|
||||
}
|
||||
|
||||
SetTimer(VoteTime, false, nameof(ConcludeVoteKick), Self);
|
||||
SetTimer(CfgKickVote.default.VoteTime, false, nameof(ConcludeVoteKick), Self);
|
||||
|
||||
RecieveVoteKick(PRI_Kicker, true);
|
||||
}
|
||||
@ -260,7 +265,7 @@ private function int VotingPlayers(optional PlayerReplicationInfo KickeePRI, opt
|
||||
foreach KFPRIs(KFPRI)
|
||||
{
|
||||
KFPC = KFPlayerController(KFPRI.Owner);
|
||||
if (KFPC != None && CVC.PlayerCanKickVote(KFPC, KFPC_Kickee))
|
||||
if (KFPC != None && KFPC != KFPC_Kickee)
|
||||
{
|
||||
VotingPlayersNum++;
|
||||
}
|
||||
@ -401,7 +406,7 @@ public reliable server function RecieveVoteKick(PlayerReplicationInfo PRI, bool
|
||||
{
|
||||
CVC.BroadcastHUDLocalized(
|
||||
CVC_KickVoteStartedHUD,
|
||||
float(VoteTime),
|
||||
float(CfgKickVote.default.VoteTime),
|
||||
KickerName,
|
||||
KickeeName,
|
||||
YesVotesPlayers);
|
||||
@ -410,7 +415,7 @@ public reliable server function RecieveVoteKick(PlayerReplicationInfo PRI, bool
|
||||
{
|
||||
CVC.BroadcastHUDLocalized(
|
||||
CVC_KickVoteReceivedHUD,
|
||||
float(VoteTime),
|
||||
float(CfgKickVote.default.VoteTime),
|
||||
YesVotesPlayers,
|
||||
NoVotesPlayers);
|
||||
}
|
||||
@ -422,7 +427,6 @@ public reliable server function RecieveVoteKick(PlayerReplicationInfo PRI, bool
|
||||
|
||||
public function bool ShouldConcludeKickVote()
|
||||
{
|
||||
local KFGameInfo KFGI;
|
||||
local int NumPRIs;
|
||||
local int KickVotesNeeded;
|
||||
|
||||
@ -433,15 +437,13 @@ public function bool ShouldConcludeKickVote()
|
||||
return Super.ShouldConcludeKickVote();
|
||||
}
|
||||
|
||||
KFGI = KFGameInfo(WorldInfo.Game);
|
||||
|
||||
NumPRIs = VotingPlayers();
|
||||
|
||||
if (YesVotes + NoVotes >= NumPRIs)
|
||||
{
|
||||
return true;
|
||||
}
|
||||
else if (KFGI != None)
|
||||
else if (GetKFGI() != None)
|
||||
{
|
||||
KickVotesNeeded = FCeil(float(NumPRIs) * KFGI.KickVotePercentage);
|
||||
KickVotesNeeded = Clamp(KickVotesNeeded, 1, NumPRIs);
|
||||
@ -465,7 +467,6 @@ public reliable server function ConcludeVoteKick()
|
||||
local KFPlayerReplicationInfo KFPRI;
|
||||
local PlayerReplicationInfo PRI;
|
||||
local int NumPRIs;
|
||||
local KFGameInfo KFGI;
|
||||
local KFPlayerController KickedPC;
|
||||
local int KickVotesNeeded;
|
||||
local int PrevKickedPlayers;
|
||||
@ -491,10 +492,8 @@ public reliable server function ConcludeVoteKick()
|
||||
{
|
||||
Super.ConcludeVoteKick();
|
||||
}
|
||||
else if (bIsKickVoteInProgress)
|
||||
else if (bIsKickVoteInProgress && GetKFGI() != None)
|
||||
{
|
||||
KFGI = KFGameInfo(WorldInfo.Game);
|
||||
|
||||
GetKFPRIArray(KFPRIs);
|
||||
|
||||
foreach KFPRIs(KFPRI) KFPRI.HideKickVote();
|
||||
@ -569,7 +568,7 @@ private function LogKickVotes()
|
||||
}
|
||||
|
||||
`Log_Kick("Kicker:" @ LogVotePlayer(Kicker));
|
||||
`Log_Kick("Kicked:" @ LogVotePlayer(Kickee) @ String(Kickee.Perk) @ String(Kickee.Level));
|
||||
`Log_Kick("Kicked:" @ LogVotePlayer(Kickee));
|
||||
|
||||
`Log_Kick("Yes voters:");
|
||||
foreach Yes(KV) `Log_Kick(LogVotePlayer(KV));
|
||||
@ -584,7 +583,110 @@ private function String LogVotePlayer(S_KickVote KV)
|
||||
{
|
||||
`Log_Trace();
|
||||
|
||||
return KV.Name @ "(UniqueID:" @ KV.UniqueID $ (KV.SteamID == "" ? "" : (", SteamID:" @ KV.SteamID $ ", Profile:" @ "https://steamcommunity.com/profiles/" $ KV.SteamID)) $ ")";
|
||||
return KV.Name @ "(UniqueID:" @ KV.UniqueID $ (KV.SteamID == "" ? "" : (", SteamID:" @ KV.SteamID $ ", Profile:" @ "https://steamcommunity.com/profiles/" $ KV.SteamID)) $ ")" @ "Perk:" @ Repl(String(KV.Perk), "KFPerk_", "", false) @ "Level:" @ String(KV.Level);
|
||||
}
|
||||
|
||||
public function ServerStartVoteSkipTrader(PlayerReplicationInfo PRI)
|
||||
{
|
||||
local Array<KFPlayerReplicationInfo> KFPRIs;
|
||||
local KFPlayerReplicationInfo KFPRI;
|
||||
local KFPlayerController KFPC;
|
||||
local byte TraderTimeRemaining;
|
||||
|
||||
KFPC = KFPlayerController(PRI.Owner);
|
||||
|
||||
if (GetKFGI() == None) return;
|
||||
|
||||
if (PRI.bOnlySpectator)
|
||||
{
|
||||
KFPC.ReceiveLocalizedMessage(class'KFLocalMessage', LMT_SkipTraderVoteNoSpectators);
|
||||
return;
|
||||
}
|
||||
|
||||
if (!bTraderIsOpen && !bForceShowSkipTrader)
|
||||
{
|
||||
KFPC.ReceiveLocalizedMessage(class'KFLocalMessage', LMT_SkipTraderIsNotOpen);
|
||||
return;
|
||||
}
|
||||
|
||||
if (bIsKickVoteInProgress || bIsPauseGameVoteInProgress)
|
||||
{
|
||||
KFPC.ReceiveLocalizedMessage(class'KFLocalMessage', LMT_OtherVoteInProgress);
|
||||
return;
|
||||
}
|
||||
|
||||
TraderTimeRemaining = GetTraderTimeRemaining();
|
||||
if(TraderTimeRemaining <= SkipTraderVoteLimit)
|
||||
{
|
||||
KFPC.ReceiveLocalizedMessage(class'KFLocalMessage', LMT_SkipTraderNoEnoughTime);
|
||||
return;
|
||||
}
|
||||
|
||||
if (!bIsSkipTraderVoteInProgress)
|
||||
{
|
||||
PlayersThatHaveVoted.Length = 0;
|
||||
|
||||
CurrentSkipTraderVote.PlayerID = PRI.UniqueId;
|
||||
CurrentSkipTraderVote.PlayerPRI = PRI;
|
||||
CurrentSkipTraderVote.PlayerIPAddress = KFPC.GetPlayerNetworkAddress();
|
||||
|
||||
bIsSkipTraderVoteInProgress = true;
|
||||
|
||||
if (bStopCountDown)
|
||||
{
|
||||
CurrentVoteTime = CfgSkipTraderVote.default.VoteTime;
|
||||
}
|
||||
else
|
||||
{
|
||||
CurrentVoteTime = Min(CfgSkipTraderVote.default.VoteTime, TraderTimeRemaining - SkipTraderVoteLimit);
|
||||
}
|
||||
|
||||
GetKFPRIArray(KFPRIs, , false);
|
||||
foreach KFPRIs(KFPRI)
|
||||
{
|
||||
KFPRI.ShowSkipTraderVote(PRI, CurrentVoteTime, !(KFPRI == PRI) && PRI.GetTeamNum() != 255);
|
||||
}
|
||||
|
||||
AllowSTPNotification = KFPRIs.Length > 1;
|
||||
|
||||
KFGI.BroadcastLocalized(KFGI, class'KFLocalMessage', LMT_SkipTraderVoteStarted, CurrentSkipTraderVote.PlayerPRI);
|
||||
SetTimer(CurrentVoteTime, false, nameof(ConcludeVoteSkipTrader), Self);
|
||||
SetTimer(1, true, nameof(UpdateTimer), Self);
|
||||
|
||||
RecieveVoteSkipTrader(PRI, true);
|
||||
|
||||
KFPlayerReplicationInfo(PRI).bAlreadyStartedASkipTraderVote = true;
|
||||
}
|
||||
else
|
||||
{
|
||||
KFPlayerController(PRI.Owner).ReceiveLocalizedMessage(class'KFLocalMessage', LMT_SkipTraderVoteInProgress);
|
||||
}
|
||||
}
|
||||
|
||||
public reliable server function UpdateTimer()
|
||||
{
|
||||
local Array<KFPlayerReplicationInfo> KFPRIs;
|
||||
local KFPlayerReplicationInfo KFPRI;
|
||||
local int VoteTimeLimit;
|
||||
|
||||
CurrentVoteTime--;
|
||||
|
||||
VoteTimeLimit = GetTraderTimeRemaining() - SkipTraderVoteLimit;
|
||||
if (!bStopCountDown && CurrentVoteTime > VoteTimeLimit)
|
||||
{
|
||||
CurrentVoteTime = VoteTimeLimit;
|
||||
}
|
||||
|
||||
GetKFPRIArray(KFPRIs, , false);
|
||||
foreach KFPRIs(KFPRI)
|
||||
{
|
||||
KFPRI.UpdateSkipTraderTime(CurrentVoteTime);
|
||||
}
|
||||
|
||||
if (CurrentVoteTime <= 0)
|
||||
{
|
||||
ConcludeVoteSkipTrader();
|
||||
}
|
||||
}
|
||||
|
||||
public reliable server function RecieveVoteSkipTrader(PlayerReplicationInfo PRI, bool bSkip)
|
||||
@ -593,7 +695,7 @@ public reliable server function RecieveVoteSkipTrader(PlayerReplicationInfo PRI,
|
||||
|
||||
`Log_Trace();
|
||||
|
||||
MustNotify = (PlayersThatHaveVoted.Find(PRI) == INDEX_NONE);
|
||||
MustNotify = (PlayersThatHaveVoted.Find(PRI) == INDEX_NONE && AllowSTPNotification);
|
||||
|
||||
Super.RecieveVoteSkipTrader(PRI, bSkip);
|
||||
|
||||
@ -611,7 +713,7 @@ public reliable server function RecieveVoteSkipTrader(PlayerReplicationInfo PRI,
|
||||
{
|
||||
CVC.BroadcastHUDLocalized(
|
||||
CVC_VoteProgressHUD,
|
||||
float(VoteTime),
|
||||
float(CfgSkipTraderVote.default.VoteTime),
|
||||
VotedPlayers(),
|
||||
DidntVotedPlayers());
|
||||
}
|
||||
@ -633,18 +735,130 @@ public reliable server function ConcludeVoteSkipTrader()
|
||||
{
|
||||
CVC.BroadcastClearMessageHUD(CfgSkipTraderVote.default.DefferedClearHUD);
|
||||
}
|
||||
|
||||
ClearTimer(nameof(ConcludeVoteSkipTrader), Self);
|
||||
ClearTimer(nameof(UpdateTimer), Self);
|
||||
}
|
||||
|
||||
Super.ConcludeVoteSkipTrader();
|
||||
}
|
||||
|
||||
public function ServerStartVotePauseGame(PlayerReplicationInfo PRI)
|
||||
{
|
||||
local Array<KFPlayerReplicationInfo> KFPRIs;
|
||||
local KFPlayerReplicationInfo KFPRI;
|
||||
local KFPlayerController KFPC;
|
||||
local byte WaveTimeRemaining;
|
||||
|
||||
if (GetKFGI() == None) return;
|
||||
|
||||
KFPC = KFPlayerController(PRI.Owner);
|
||||
|
||||
if (PRI.bOnlySpectator)
|
||||
{
|
||||
KFPC.ReceiveLocalizedMessage(class'KFLocalMessage', bIsEndlessPaused ? LMT_ResumeVoteNoSpectators : LMT_PauseVoteNoSpectators);
|
||||
return;
|
||||
}
|
||||
|
||||
if (bWaveIsActive)
|
||||
{
|
||||
KFPC.ReceiveLocalizedMessage(class'KFLocalMessage', bIsEndlessPaused ? LMT_ResumeVoteWaveActive : LMT_PauseVoteWaveActive);
|
||||
return;
|
||||
}
|
||||
|
||||
if (!bEndlessMode)
|
||||
{
|
||||
KFPC.ReceiveLocalizedMessage(class'KFLocalMessage', LMT_PauseVoteWrongMode);
|
||||
return;
|
||||
}
|
||||
|
||||
if (bIsKickVoteInProgress || bIsSkipTraderVoteInProgress)
|
||||
{
|
||||
KFPC.ReceiveLocalizedMessage(class'KFLocalMessage', LMT_OtherVoteInProgress);
|
||||
return;
|
||||
}
|
||||
|
||||
WaveTimeRemaining = GetTraderTimeRemaining();
|
||||
if (WaveTimeRemaining <= PauseGameVoteLimit)
|
||||
{
|
||||
KFPC.ReceiveLocalizedMessage(class'KFLocalMessage', bIsEndlessPaused ? LMT_ResumeVoteNoEnoughTime : LMT_PauseVoteNoEnoughTime);
|
||||
return;
|
||||
}
|
||||
|
||||
if (!bIsPauseGameVoteInProgress)
|
||||
{
|
||||
PlayersThatHaveVoted.Length = 0;
|
||||
|
||||
CurrentPauseGameVote.PlayerID = PRI.UniqueId;
|
||||
CurrentPauseGameVote.PlayerPRI = PRI;
|
||||
CurrentPauseGameVote.PlayerIPAddress = KFPC.GetPlayerNetworkAddress();
|
||||
|
||||
bIsPauseGameVoteInProgress = true;
|
||||
|
||||
if (bStopCountDown)
|
||||
{
|
||||
CurrentVoteTime = CfgPauseVote.default.VoteTime;
|
||||
}
|
||||
else
|
||||
{
|
||||
CurrentVoteTime = Min(CfgPauseVote.default.VoteTime, WaveTimeRemaining - PauseGameVoteLimit);
|
||||
}
|
||||
|
||||
GetKFPRIArray(KFPRIs);
|
||||
foreach KFPRIs(KFPRI)
|
||||
{
|
||||
KFPRI.ShowPauseGameVote(PRI, CurrentVoteTime, !(KFPRI == PRI));
|
||||
}
|
||||
|
||||
AllowSTPNotification = KFPRIs.Length > 1;
|
||||
|
||||
KFGI.BroadcastLocalized(KFGI, class'KFLocalMessage', bIsEndlessPaused ? LMT_ResumeVoteStarted : LMT_PauseVoteStarted, CurrentPauseGameVote.PlayerPRI);
|
||||
SetTimer(CurrentVoteTime, false, nameof(ConcludeVotePauseGame), Self);
|
||||
SetTimer(1, true, nameof(UpdatePauseGameTimer), Self);
|
||||
|
||||
ReceiveVotePauseGame(PRI, true);
|
||||
|
||||
KFPlayerReplicationInfo(PRI).bAlreadyStartedAPauseGameVote = true;
|
||||
}
|
||||
else
|
||||
{
|
||||
KFPlayerController(PRI.Owner).ReceiveLocalizedMessage(class'KFLocalMessage', bIsEndlessPaused ? LMT_ResumeVoteInProgress : LMT_PauseVoteInProgress);
|
||||
}
|
||||
}
|
||||
|
||||
public reliable server function UpdatePauseGameTimer() // TODO:
|
||||
{
|
||||
local Array<KFPlayerReplicationInfo> KFPRIs;
|
||||
local KFPlayerReplicationInfo KFPRI;
|
||||
local int VoteTimeLimit;
|
||||
|
||||
CurrentVoteTime--;
|
||||
|
||||
VoteTimeLimit = GetTraderTimeRemaining() - PauseGameVoteLimit;
|
||||
if (!bStopCountDown && CurrentVoteTime > VoteTimeLimit)
|
||||
{
|
||||
CurrentVoteTime = VoteTimeLimit;
|
||||
}
|
||||
|
||||
GetKFPRIArray(KFPRIs);
|
||||
foreach KFPRIs(KFPRI)
|
||||
{
|
||||
KFPRI.UpdatePauseGameTime(CurrentVoteTime);
|
||||
}
|
||||
|
||||
if (CurrentVoteTime <= 0)
|
||||
{
|
||||
ConcludeVotePauseGame();
|
||||
}
|
||||
}
|
||||
|
||||
public reliable server function ReceiveVotePauseGame(PlayerReplicationInfo PRI, bool bSkip)
|
||||
{
|
||||
local bool MustNotify;
|
||||
|
||||
`Log_Trace();
|
||||
|
||||
MustNotify = (PlayersThatHaveVoted.Find(PRI) == INDEX_NONE);
|
||||
MustNotify = (PlayersThatHaveVoted.Find(PRI) == INDEX_NONE && AllowSTPNotification);
|
||||
|
||||
Super.ReceiveVotePauseGame(PRI, bSkip);
|
||||
|
||||
@ -662,7 +876,7 @@ public reliable server function ReceiveVotePauseGame(PlayerReplicationInfo PRI,
|
||||
{
|
||||
CVC.BroadcastHUDLocalized(
|
||||
CVC_VoteProgressHUD,
|
||||
float(VoteTime),
|
||||
float(CfgPauseVote.default.VoteTime),
|
||||
VotedPlayers(),
|
||||
DidntVotedPlayers());
|
||||
}
|
||||
@ -684,6 +898,9 @@ public reliable server function ConcludeVotePauseGame()
|
||||
{
|
||||
CVC.BroadcastClearMessageHUD(CfgPauseVote.default.DefferedClearHUD);
|
||||
}
|
||||
|
||||
ClearTimer(nameof(ConcludeVotePauseGame), Self);
|
||||
ClearTimer(nameof(UpdatePauseGameTimer), Self);
|
||||
}
|
||||
|
||||
Super.ConcludeVotePauseGame();
|
||||
@ -691,8 +908,6 @@ public reliable server function ConcludeVotePauseGame()
|
||||
|
||||
private function Array<String> ActiveMapCycle()
|
||||
{
|
||||
local KFGameInfo KFGI;
|
||||
|
||||
`Log_Trace();
|
||||
|
||||
if (WorldInfo.NetMode == NM_Standalone)
|
||||
@ -700,8 +915,7 @@ private function Array<String> ActiveMapCycle()
|
||||
return Maplist;
|
||||
}
|
||||
|
||||
KFGI = KFGameInfo(WorldInfo.Game);
|
||||
if (KFGI != None)
|
||||
if (GetKFGI() != None)
|
||||
{
|
||||
return KFGI.GameMapCycles[KFGI.ActiveMapCycle].Maps;
|
||||
}
|
||||
@ -712,14 +926,12 @@ private function Array<String> GetAviableMaps()
|
||||
local String LowerDefaultNextMap;
|
||||
local Array<String> MapCycle;
|
||||
local Array<String> Maps;
|
||||
local KFGameInfo KFGI;
|
||||
local String Map;
|
||||
local int Index;
|
||||
|
||||
`Log_Trace();
|
||||
|
||||
KFGI = KFGameInfo(WorldInfo.Game);
|
||||
if (KFGI == None) return Maps;
|
||||
if (GetKFGI() == None) return Maps;
|
||||
|
||||
MapCycle = ActiveMapCycle();
|
||||
|
||||
@ -797,16 +1009,12 @@ private function bool IsCustomMap(String MapName)
|
||||
|
||||
private function int DefaultNextMapIndex()
|
||||
{
|
||||
local KFGameInfo KFGI;
|
||||
local Array<String> AviableMaps;
|
||||
local Array<String> MapCycle;
|
||||
local int CurrentMapIndex;
|
||||
|
||||
`Log_Trace();
|
||||
|
||||
KFGI = KFGameInfo(WorldInfo.Game);
|
||||
if (KFGI == None) return INDEX_NONE;
|
||||
|
||||
MapCycle = ActiveMapCycle();
|
||||
AviableMaps = GetAviableMaps();
|
||||
|
||||
@ -915,4 +1123,5 @@ public function int GetNextMap()
|
||||
defaultproperties
|
||||
{
|
||||
AllowHudNotification = true;
|
||||
AllowSTPNotification = true;
|
||||
}
|
||||
|
@ -60,7 +60,7 @@ private static function bool IsUID(String ID, E_LogLevel LogLevel)
|
||||
{
|
||||
`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)
|
||||
|
@ -64,7 +64,7 @@ static function IncMapStat(String Map, int PlayTime, String SortPolicy, E_LogLev
|
||||
StaticSaveConfig();
|
||||
}
|
||||
|
||||
DefaultProperties
|
||||
defaultproperties
|
||||
{
|
||||
|
||||
}
|
||||
|
60
CVC/Classes/Mut.uc
Normal file
60
CVC/Classes/Mut.uc
Normal 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")
|
||||
}
|
BIN
Localization/CHN/CVC.chn
Normal file
BIN
Localization/CHN/CVC.chn
Normal file
Binary file not shown.
BIN
Localization/CHT/CVC.cht
Normal file
BIN
Localization/CHT/CVC.cht
Normal file
Binary file not shown.
Binary file not shown.
Binary file not shown.
@ -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]
|
||||
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]
|
||||
|
||||
[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]
|
||||
[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):
|
||||
[b]ServerSubscribedWorkshopItems=2847465899[/b]
|
||||
[*]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]
|
||||
|
||||
[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]
|
||||
[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]NegativeColorHex[/b] - hex color for no vote in chat;
|
||||
[*][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]
|
||||
|
||||
[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]NegativeColorHex[/b] - hex color for no vote in chat;
|
||||
[*][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]
|
||||
|
||||
[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]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]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]
|
||||
|
||||
[b][CVC.KickProtected][/b]
|
||||
@ -104,18 +110,29 @@ This section sets the next map when no one voted for the map.
|
||||
[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).
|
||||
[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.
|
||||
[/list]
|
||||
|
||||
[b]How start wave kick protection works:[/b]
|
||||
When a player joins a server, he is protected from a kick for the specified number of [b]Waves[/b]. This keeps the server from being taken over by players, and it also forces current players to play with the new player for at least a little bit before they can kick him. This solves most of the unfair kicks in the game.
|
||||
|
||||
Along with receiving the kick protection, the new player loses the ability to vote for the kick. This eliminates the ability for new players to remove existing players using kick protection for impunity.
|
||||
Along with receiving the kick protection, the new player loses the ability to start kick vote. This eliminates the ability for new players to remove existing players using kick protection for impunity.
|
||||
|
||||
When the player has played the specified number of [b]Waves[/b], he loses the kick protection and gets the opportunity to vote for the kick.
|
||||
When the player has played the specified number of [b]Waves[/b], he loses the kick protection and gets the opportunity to start kick vote.
|
||||
|
||||
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]
|
||||
[url=https://github.com/GenZmeY/KF2-ControlledVoteCollector]https://github.com/GenZmeY/KF2-ControlledVoteCollector[/url] [b](GNU GPLv3)[/b]
|
18
README.md
18
README.md
@ -1,16 +1,16 @@
|
||||
# Controlled Vote Collector
|
||||
|
||||
[](https://steamcommunity.com/sharedfiles/filedetails/?id=2847465899)
|
||||
[](https://steamcommunity.com/sharedfiles/filedetails/?id=2847465899)
|
||||
[](https://steamcommunity.com/sharedfiles/filedetails/?id=2848836389)
|
||||
[](https://steamcommunity.com/sharedfiles/filedetails/?id=2847465899)
|
||||
[](https://steamcommunity.com/sharedfiles/filedetails/?id=2847465899)
|
||||
[](https://github.com/GenZmeY/KF2-ControlledVoteCollector/actions/workflows/mega-linter.yml)
|
||||
[](https://github.com/GenZmeY/KF2-ControlledVoteCollector/tags)
|
||||
[](LICENSE)
|
||||
|
||||
# Description
|
||||
## Description
|
||||
New vote collector with improvements and features.
|
||||
|
||||
# Features
|
||||
## Features
|
||||
- map statistics;
|
||||
- default/next map setting;
|
||||
- anonymous or public voting;
|
||||
@ -18,10 +18,10 @@ New vote collector with improvements and features.
|
||||
- kick voting setup;
|
||||
- early kick protection.
|
||||
|
||||
# Usage & Setup
|
||||
## Usage & Setup
|
||||
[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.
|
||||
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:
|
||||
@ -35,9 +35,9 @@ New vote collector with improvements and features.
|
||||
5. The compiled files will be here:
|
||||
`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.
|
||||
Describe what the bug looks like and how reproduce it.
|
||||
|
||||
# License
|
||||
[GNU GPLv3](LICENSE)
|
||||
## License
|
||||
[](LICENSE)
|
||||
|
@ -11,6 +11,15 @@ StripSource="True"
|
||||
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 ###
|
||||
|
||||
# Mutators that will be uploaded to the workshop
|
||||
|
2
tools
2
tools
@ -1 +1 @@
|
||||
Subproject commit 2f173aad7a6f4578574764801136a0d86e830653
|
||||
Subproject commit fb458ac61f7e6c6426b8dff366dd5e7499e0d95f
|
Loading…
x
Reference in New Issue
Block a user