Compare commits
23 Commits
Author | SHA1 | Date | |
---|---|---|---|
bf5e6e7b9c | |||
bb8225bc8f | |||
e6ed3a4ebf | |||
85e235316c | |||
10bd48e082 | |||
d55cc77e05 | |||
7e2351c1c1 | |||
6344c3b3e3 | |||
e0f6ab61b0 | |||
2b45b7ae56 | |||
9ab02e29a1 | |||
f66b991b26 | |||
05e2ca250c | |||
281b10e427 | |||
728e0b3be3 | |||
335ca77980 | |||
1af2ef0b80 | |||
f1f6478548 | |||
3ffd4dcd15 | |||
07a339f188 | |||
eb3965e68a | |||
36a61fd084 | |||
7668a07a18 |
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
|
114
.github/workflows/mega-linter.yml
vendored
Normal file
114
.github/workflows/mega-linter.yml
vendored
Normal file
@ -0,0 +1,114 @@
|
|||||||
|
---
|
||||||
|
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
|
||||||
|
|
||||||
|
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"
|
@ -1,238 +1,235 @@
|
|||||||
class LTI extends Info
|
class LTI extends Info
|
||||||
config(LTI);
|
config(LTI);
|
||||||
|
|
||||||
const LatestVersion = 2;
|
const LatestVersion = 2;
|
||||||
|
|
||||||
const CfgRemoveItems = class'RemoveItems';
|
const CfgRemoveItems = class'RemoveItems';
|
||||||
const CfgOfficialWeapons = class'OfficialWeapons';
|
const CfgOfficialWeapons = class'OfficialWeapons';
|
||||||
const Trader = class'Trader';
|
const Trader = class'Trader';
|
||||||
|
|
||||||
var private config int Version;
|
var private config int Version;
|
||||||
var private config E_LogLevel LogLevel;
|
var private config E_LogLevel LogLevel;
|
||||||
var private config bool bOfficialWeaponsList;
|
var private config bool bOfficialWeaponsList;
|
||||||
|
|
||||||
var private KFGameInfo KFGI;
|
var private KFGameInfo KFGI;
|
||||||
var private KFGameReplicationInfo KFGRI;
|
var private KFGameReplicationInfo KFGRI;
|
||||||
|
|
||||||
var private Array<class<KFWeaponDefinition> > RemoveItems;
|
var private Array<class<KFWeaponDefinition> > WeapDefs;
|
||||||
var private Array<LTI_RepInfo> RepInfos;
|
var private Array<class<KFWeaponDefinition> > RemoveItems;
|
||||||
var private bool ReadyToSync;
|
var private Array<LTI_RepInfo> RepInfos;
|
||||||
|
var private bool ReadyToSync;
|
||||||
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();
|
||||||
`Log_Debug("PreBeginPlay readyToSync" @ ReadyToSync);
|
|
||||||
|
`Log_Debug("PreBeginPlay readyToSync" @ ReadyToSync);
|
||||||
if (WorldInfo.NetMode == NM_Client)
|
|
||||||
{
|
if (WorldInfo.NetMode == NM_Client)
|
||||||
`Log_Fatal("NetMode == NM_Client, Destroy...");
|
{
|
||||||
SafeDestroy();
|
`Log_Fatal("NetMode == NM_Client, Destroy...");
|
||||||
return;
|
SafeDestroy();
|
||||||
}
|
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;
|
{
|
||||||
SaveConfig();
|
LogLevel = LL_Info;
|
||||||
}
|
SaveConfig();
|
||||||
|
}
|
||||||
CfgRemoveItems.static.InitConfig(Version, LatestVersion);
|
|
||||||
|
CfgRemoveItems.static.InitConfig(Version, LatestVersion);
|
||||||
switch (Version)
|
|
||||||
{
|
switch (Version)
|
||||||
case `NO_CONFIG:
|
{
|
||||||
`Log_Info("Config created");
|
case `NO_CONFIG:
|
||||||
|
`Log_Info("Config created");
|
||||||
case 1:
|
|
||||||
|
case 1:
|
||||||
case MaxInt:
|
|
||||||
`Log_Info("Config updated to version" @ LatestVersion);
|
case MaxInt:
|
||||||
break;
|
`Log_Info("Config updated to version" @ LatestVersion);
|
||||||
|
break;
|
||||||
case LatestVersion:
|
|
||||||
`Log_Info("Config is up-to-date");
|
case LatestVersion:
|
||||||
break;
|
`Log_Info("Config is up-to-date");
|
||||||
|
break;
|
||||||
default:
|
|
||||||
`Log_Warn("The config version is higher than the current version (are you using an old mutator?)");
|
default:
|
||||||
`Log_Warn("Config version is" @ Version @ "but current version is" @ LatestVersion);
|
`Log_Warn("The config version is higher than the current version (are you using an old mutator?)");
|
||||||
`Log_Warn("The config version will be changed to" @ LatestVersion);
|
`Log_Warn("Config version is" @ Version @ "but current version is" @ LatestVersion);
|
||||||
break;
|
`Log_Warn("The config version will be changed to" @ LatestVersion);
|
||||||
}
|
break;
|
||||||
|
}
|
||||||
CfgOfficialWeapons.static.Update(bOfficialWeaponsList);
|
|
||||||
|
CfgOfficialWeapons.static.Update(bOfficialWeaponsList);
|
||||||
if (LatestVersion != Version)
|
|
||||||
{
|
if (LatestVersion != Version)
|
||||||
Version = LatestVersion;
|
{
|
||||||
SaveConfig();
|
Version = LatestVersion;
|
||||||
}
|
SaveConfig();
|
||||||
|
}
|
||||||
if (LogLevel == LL_WrongLevel)
|
|
||||||
{
|
if (LogLevel == LL_WrongLevel)
|
||||||
LogLevel = LL_Info;
|
{
|
||||||
`Log_Warn("Wrong 'LogLevel', return to default value");
|
LogLevel = LL_Info;
|
||||||
SaveConfig();
|
`Log_Warn("Wrong 'LogLevel', return to default value");
|
||||||
}
|
SaveConfig();
|
||||||
`Log_Base("LogLevel:" @ LogLevel);
|
}
|
||||||
|
`Log_Base("LogLevel:" @ LogLevel);
|
||||||
RemoveItems = CfgRemoveItems.static.Load(LogLevel);
|
|
||||||
}
|
RemoveItems = CfgRemoveItems.static.Load(LogLevel);
|
||||||
|
}
|
||||||
private function PostInit()
|
|
||||||
{
|
private function PostInit()
|
||||||
local LTI_RepInfo RepInfo;
|
{
|
||||||
|
local LTI_RepInfo RepInfo;
|
||||||
`Log_Trace();
|
|
||||||
|
`Log_Trace();
|
||||||
if (WorldInfo == None || WorldInfo.Game == None)
|
|
||||||
{
|
if (WorldInfo == None || WorldInfo.Game == None)
|
||||||
SetTimer(1.0f, false, nameof(PostInit));
|
{
|
||||||
return;
|
SetTimer(1.0f, false, nameof(PostInit));
|
||||||
}
|
return;
|
||||||
|
}
|
||||||
KFGI = KFGameInfo(WorldInfo.Game);
|
|
||||||
if (KFGI == None)
|
KFGI = KFGameInfo(WorldInfo.Game);
|
||||||
{
|
if (KFGI == None)
|
||||||
`Log_Fatal("Incompatible gamemode:" @ WorldInfo.Game);
|
{
|
||||||
SafeDestroy();
|
`Log_Fatal("Incompatible gamemode:" @ WorldInfo.Game);
|
||||||
return;
|
SafeDestroy();
|
||||||
}
|
return;
|
||||||
|
}
|
||||||
if (KFGI.GameReplicationInfo == None)
|
|
||||||
{
|
if (KFGI.GameReplicationInfo == None)
|
||||||
SetTimer(1.0f, false, nameof(PostInit));
|
{
|
||||||
return;
|
SetTimer(1.0f, false, nameof(PostInit));
|
||||||
}
|
return;
|
||||||
|
}
|
||||||
KFGRI = KFGameReplicationInfo(KFGI.GameReplicationInfo);
|
|
||||||
if (KFGRI == None)
|
KFGRI = KFGameReplicationInfo(KFGI.GameReplicationInfo);
|
||||||
{
|
if (KFGRI == None)
|
||||||
`Log_Fatal("Incompatible Replication info:" @ KFGI.GameReplicationInfo);
|
{
|
||||||
SafeDestroy();
|
`Log_Fatal("Incompatible Replication info:" @ KFGI.GameReplicationInfo);
|
||||||
return;
|
SafeDestroy();
|
||||||
}
|
return;
|
||||||
|
}
|
||||||
Trader.static.ModifyTrader(
|
|
||||||
KFGRI,
|
Trader.static.ModifyTrader(
|
||||||
RemoveItems,
|
KFGRI,
|
||||||
CfgRemoveItems.default.bAll,
|
RemoveItems,
|
||||||
CfgRemoveItems.default.bHRG,
|
CfgRemoveItems.default.bAll,
|
||||||
CfgRemoveItems.default.bDLC,
|
CfgRemoveItems.default.bHRG,
|
||||||
LogLevel);
|
CfgRemoveItems.default.bDLC,
|
||||||
|
LogLevel);
|
||||||
ReadyToSync = true;
|
|
||||||
|
WeapDefs = Trader.static.GetTraderWeapDefs(KFGRI, LogLevel);
|
||||||
foreach RepInfos(RepInfo)
|
|
||||||
{
|
ReadyToSync = true;
|
||||||
if (RepInfo.PendingSync)
|
|
||||||
{
|
foreach RepInfos(RepInfo)
|
||||||
RepInfo.ServerSync();
|
{
|
||||||
}
|
if (RepInfo.PendingSync)
|
||||||
}
|
{
|
||||||
}
|
RepInfo.Replicate(WeapDefs);
|
||||||
|
}
|
||||||
public function NotifyLogin(Controller C)
|
}
|
||||||
{
|
}
|
||||||
`Log_Trace();
|
|
||||||
|
public function NotifyLogin(Controller C)
|
||||||
if (!CreateRepInfo(C))
|
{
|
||||||
{
|
`Log_Trace();
|
||||||
`Log_Error("Can't create RepInfo for:" @ C);
|
|
||||||
}
|
if (!CreateRepInfo(C))
|
||||||
}
|
{
|
||||||
|
`Log_Error("Can't create RepInfo for:" @ C);
|
||||||
public function NotifyLogout(Controller C)
|
}
|
||||||
{
|
}
|
||||||
`Log_Trace();
|
|
||||||
|
public function NotifyLogout(Controller C)
|
||||||
DestroyRepInfo(C);
|
{
|
||||||
}
|
`Log_Trace();
|
||||||
|
|
||||||
public function bool CreateRepInfo(Controller C)
|
DestroyRepInfo(C);
|
||||||
{
|
}
|
||||||
local LTI_RepInfo RepInfo;
|
|
||||||
|
public function bool CreateRepInfo(Controller C)
|
||||||
`Log_Trace();
|
{
|
||||||
|
local LTI_RepInfo RepInfo;
|
||||||
if (C == None) return false;
|
|
||||||
|
`Log_Trace();
|
||||||
RepInfo = Spawn(class'LTI_RepInfo', C);
|
|
||||||
|
if (C == None) return false;
|
||||||
if (RepInfo == None) return false;
|
|
||||||
|
RepInfo = Spawn(class'LTI_RepInfo', C);
|
||||||
RepInfo.PrepareSync(
|
|
||||||
Self,
|
if (RepInfo == None) return false;
|
||||||
LogLevel,
|
|
||||||
RemoveItems,
|
RepInfo.PrepareSync(Self, LogLevel);
|
||||||
CfgRemoveItems.default.bAll,
|
|
||||||
CfgRemoveItems.default.bHRG,
|
RepInfos.AddItem(RepInfo);
|
||||||
CfgRemoveItems.default.bDLC);
|
|
||||||
|
if (ReadyToSync)
|
||||||
RepInfos.AddItem(RepInfo);
|
{
|
||||||
|
RepInfo.Replicate(WeapDefs);
|
||||||
if (ReadyToSync)
|
}
|
||||||
{
|
else
|
||||||
RepInfo.ServerSync();
|
{
|
||||||
}
|
RepInfo.PendingSync = true;
|
||||||
else
|
}
|
||||||
{
|
|
||||||
RepInfo.PendingSync = true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
public function bool DestroyRepInfo(Controller C)
|
||||||
}
|
{
|
||||||
|
local LTI_RepInfo RepInfo;
|
||||||
public function bool DestroyRepInfo(Controller C)
|
|
||||||
{
|
`Log_Trace();
|
||||||
local LTI_RepInfo RepInfo;
|
|
||||||
|
if (C == None) return false;
|
||||||
`Log_Trace();
|
|
||||||
|
foreach RepInfos(RepInfo)
|
||||||
if (C == None) return false;
|
{
|
||||||
|
if (RepInfo.Owner == C)
|
||||||
foreach RepInfos(RepInfo)
|
{
|
||||||
{
|
RepInfos.RemoveItem(RepInfo);
|
||||||
if (RepInfo.Owner == C)
|
RepInfo.SafeDestroy();
|
||||||
{
|
return true;
|
||||||
RepInfos.RemoveItem(RepInfo);
|
}
|
||||||
RepInfo.SafeDestroy();
|
}
|
||||||
return true;
|
|
||||||
}
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
DefaultProperties
|
||||||
}
|
{
|
||||||
|
ReadyToSync = false
|
||||||
DefaultProperties
|
|
||||||
{
|
|
||||||
ReadyToSync = false
|
|
||||||
}
|
}
|
@ -1,4 +1,4 @@
|
|||||||
[Flags]
|
[Flags]
|
||||||
AllowDownload=True
|
AllowDownload=True
|
||||||
ClientOptional=False
|
ClientOptional=False
|
||||||
ServerSideOnly=False
|
ServerSideOnly=False
|
||||||
|
@ -1,60 +1 @@
|
|||||||
class LTIMut extends KFMutator;
|
class LTIMut extends Mut; // backward compatibility
|
||||||
|
|
||||||
var private LTI LTI;
|
|
||||||
|
|
||||||
public simulated function bool SafeDestroy()
|
|
||||||
{
|
|
||||||
return (bPendingDelete || bDeleteMe || Destroy());
|
|
||||||
}
|
|
||||||
|
|
||||||
public event PreBeginPlay()
|
|
||||||
{
|
|
||||||
Super.PreBeginPlay();
|
|
||||||
|
|
||||||
if (WorldInfo.NetMode == NM_Client) return;
|
|
||||||
|
|
||||||
foreach WorldInfo.DynamicActors(class'LTI', LTI)
|
|
||||||
{
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (LTI == None)
|
|
||||||
{
|
|
||||||
LTI = WorldInfo.Spawn(class'LTI');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (LTI == None)
|
|
||||||
{
|
|
||||||
`Log_Base("FATAL: Can't Spawn 'LTI'");
|
|
||||||
SafeDestroy();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public function AddMutator(Mutator Mut)
|
|
||||||
{
|
|
||||||
if (Mut == Self) return;
|
|
||||||
|
|
||||||
if (Mut.Class == Class)
|
|
||||||
LTIMut(Mut).SafeDestroy();
|
|
||||||
else
|
|
||||||
Super.AddMutator(Mut);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function NotifyLogin(Controller C)
|
|
||||||
{
|
|
||||||
LTI.NotifyLogin(C);
|
|
||||||
|
|
||||||
Super.NotifyLogin(C);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function NotifyLogout(Controller C)
|
|
||||||
{
|
|
||||||
LTI.NotifyLogout(C);
|
|
||||||
|
|
||||||
Super.NotifyLogout(C);
|
|
||||||
}
|
|
||||||
|
|
||||||
DefaultProperties
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
|
@ -1,73 +1,89 @@
|
|||||||
class LTI_LocalMessage extends Object
|
class LTI_LocalMessage extends Object
|
||||||
abstract;
|
abstract;
|
||||||
|
|
||||||
var const String SyncItemsDefault;
|
var const String SyncItemsDefault;
|
||||||
var private localized String SyncItems;
|
var private localized String SyncItems;
|
||||||
|
|
||||||
var const String SyncFinishedDefault;
|
var const String WaitingGRIDefault;
|
||||||
var private localized String SyncFinished;
|
var private localized String WaitingGRI;
|
||||||
|
|
||||||
var const String WaitingGRIDefault;
|
var const String IncompatibleGRIDefault;
|
||||||
var private localized String WaitingGRI;
|
var private localized String IncompatibleGRI;
|
||||||
|
|
||||||
var const String IncompatibleGRIDefault;
|
var const String IncompatibleGRIWarningDefault;
|
||||||
var private localized String IncompatibleGRI;
|
var private localized String IncompatibleGRIWarning;
|
||||||
|
|
||||||
var const String DisconnectDefault;
|
var const String NoneGRIDefault;
|
||||||
var private localized String Disconnect;
|
var private localized String NoneGRI;
|
||||||
|
|
||||||
var const String SecondsShortDefault;
|
var const String NoneGRIWarningDefault;
|
||||||
var private localized String SecondsShort;
|
var private localized String NoneGRIWarning;
|
||||||
|
|
||||||
enum E_LTI_LocalMessageType
|
var const String SecondsShortDefault;
|
||||||
{
|
var private localized String SecondsShort;
|
||||||
LTI_SyncItems,
|
|
||||||
LTI_SyncFinished,
|
var const String PleaseWaitDefault;
|
||||||
LTI_WaitingGRI,
|
var private localized String PleaseWait;
|
||||||
LTI_IncompatibleGRI,
|
|
||||||
LTI_Disconnect,
|
enum E_LTI_LocalMessageType
|
||||||
LTI_SecondsShort
|
{
|
||||||
};
|
LTI_SyncItems,
|
||||||
|
LTI_WaitingGRI,
|
||||||
public static function String GetLocalizedString(
|
LTI_IncompatibleGRI,
|
||||||
E_LogLevel LogLevel,
|
LTI_IncompatibleGRIWarning,
|
||||||
E_LTI_LocalMessageType LMT,
|
LTI_NoneGRI,
|
||||||
optional String String1,
|
LTI_NoneGRIWarning,
|
||||||
optional String String2,
|
LTI_SecondsShort,
|
||||||
optional String String3)
|
LTI_PleaseWait
|
||||||
{
|
};
|
||||||
`Log_TraceStatic();
|
|
||||||
|
public static function String GetLocalizedString(
|
||||||
switch (LMT)
|
E_LogLevel LogLevel,
|
||||||
{
|
E_LTI_LocalMessageType LMT,
|
||||||
case LTI_SyncItems:
|
optional String String1,
|
||||||
return (default.SyncItems != "" ? default.SyncItems : default.SyncItemsDefault);
|
optional String String2,
|
||||||
|
optional String String3)
|
||||||
case LTI_SyncFinished:
|
{
|
||||||
return (default.SyncFinished != "" ? default.SyncFinished : default.SyncFinishedDefault);
|
`Log_TraceStatic();
|
||||||
|
|
||||||
case LTI_WaitingGRI:
|
switch (LMT)
|
||||||
return (default.WaitingGRI != "" ? default.WaitingGRI : default.WaitingGRIDefault);
|
{
|
||||||
|
case LTI_SyncItems:
|
||||||
case LTI_IncompatibleGRI:
|
return (default.SyncItems != "" ? default.SyncItems : default.SyncItemsDefault);
|
||||||
return (default.IncompatibleGRI != "" ? default.IncompatibleGRI : default.IncompatibleGRIDefault);
|
|
||||||
|
case LTI_WaitingGRI:
|
||||||
case LTI_Disconnect:
|
return (default.WaitingGRI != "" ? default.WaitingGRI : default.WaitingGRIDefault);
|
||||||
return (default.Disconnect != "" ? default.Disconnect : default.DisconnectDefault);
|
|
||||||
|
case LTI_IncompatibleGRI:
|
||||||
case LTI_SecondsShort:
|
return (default.IncompatibleGRI != "" ? default.IncompatibleGRI : default.IncompatibleGRIDefault) @ String1;
|
||||||
return (default.SecondsShort != "" ? default.SecondsShort : default.SecondsShortDefault);
|
|
||||||
}
|
case LTI_IncompatibleGRIWarning:
|
||||||
|
return (default.IncompatibleGRIWarning != "" ? default.IncompatibleGRIWarning : default.IncompatibleGRIWarningDefault);
|
||||||
return "";
|
|
||||||
}
|
case LTI_NoneGRI:
|
||||||
|
return (default.NoneGRI != "" ? default.NoneGRI : default.NoneGRIDefault);
|
||||||
defaultproperties
|
|
||||||
{
|
case LTI_NoneGRIWarning:
|
||||||
SyncItemsDefault = "Sync items:"
|
return (default.NoneGRIWarning != "" ? default.NoneGRIWarning : default.NoneGRIWarningDefault);
|
||||||
SyncFinishedDefault = "Sync finished."
|
|
||||||
WaitingGRIDefault = "Waiting GRI..."
|
case LTI_SecondsShort:
|
||||||
IncompatibleGRIDefault = "Incompatible GRI:"
|
return (default.SecondsShort != "" ? default.SecondsShort : default.SecondsShortDefault);
|
||||||
DisconnectDefault = "Disconnect..."
|
|
||||||
SecondsShortDefault = "s"
|
case LTI_PleaseWait:
|
||||||
|
return (default.PleaseWait != "" ? default.PleaseWait : default.PleaseWaitDefault);
|
||||||
|
}
|
||||||
|
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
defaultproperties
|
||||||
|
{
|
||||||
|
SyncItemsDefault = "Sync items:"
|
||||||
|
WaitingGRIDefault = "Waiting GRI..."
|
||||||
|
IncompatibleGRIDefault = "Incompatible GRI:"
|
||||||
|
IncompatibleGRIWarningDefault = "You can enter the game, but the trader may not work correctly.";
|
||||||
|
NoneGRIDefault = "GRI is not initialized!"
|
||||||
|
NoneGRIWarningDefault = "It is recommended to reconnect. If you enter the game right now, the trader may not work correctly.";
|
||||||
|
SecondsShortDefault = "s"
|
||||||
|
PleaseWaitDefault = "Please wait"
|
||||||
}
|
}
|
@ -1,282 +1,396 @@
|
|||||||
class LTI_RepInfo extends ReplicationInfo;
|
class LTI_RepInfo extends ReplicationInfo;
|
||||||
|
|
||||||
const Trader = class'Trader';
|
const CAPACITY = 64; // max: 128
|
||||||
const LocalMessage = class'LTI_LocalMessage';
|
|
||||||
|
const Trader = class'Trader';
|
||||||
var public bool PendingSync;
|
const LocalMessage = class'LTI_LocalMessage';
|
||||||
|
|
||||||
var private LTI LTI;
|
struct ReplicationStruct
|
||||||
var private E_LogLevel LogLevel;
|
{
|
||||||
var private Array<class<KFWeaponDefinition> > RemoveItems;
|
var int Size;
|
||||||
var private bool ReplaceMode;
|
var int Transfered;
|
||||||
var private bool RemoveHRG;
|
|
||||||
var private bool RemoveDLC;
|
var class<KFWeaponDefinition> Items[CAPACITY];
|
||||||
|
var int Length;
|
||||||
var private int Recieved;
|
};
|
||||||
var private int SyncSize;
|
|
||||||
|
var public bool PendingSync;
|
||||||
var private KFPlayerController KFPC;
|
|
||||||
var private KFGFxWidget_PartyInGame PartyInGameWidget;
|
var private LTI LTI;
|
||||||
var private GFxObject Notification;
|
var private E_LogLevel LogLevel;
|
||||||
|
|
||||||
var private String NotificationHeaderText;
|
var private GameReplicationInfo GRI;
|
||||||
var private String NotificationLeftText;
|
var private KFPlayerController KFPC;
|
||||||
var private String NotificationRightText;
|
var private KFGFxWidget_PartyInGame PartyInGameWidget;
|
||||||
var private int NotificationPercent;
|
var private GFxObject Notification;
|
||||||
|
|
||||||
var private int WaitingGRI;
|
var private String NotificationHeaderText;
|
||||||
|
var private String NotificationLeftText;
|
||||||
replication
|
var private String NotificationRightText;
|
||||||
{
|
var private int NotificationPercent;
|
||||||
if (bNetInitial && Role == ROLE_Authority)
|
|
||||||
LogLevel, ReplaceMode, RemoveHRG, RemoveDLC, SyncSize;
|
var private int WaitingGRI;
|
||||||
}
|
var private int WaitingGRIThreshold;
|
||||||
|
var private int WaitingGRILimit;
|
||||||
public simulated function bool SafeDestroy()
|
|
||||||
{
|
var private ReplicationStruct RepData;
|
||||||
`Log_Trace();
|
var private Array<class<KFWeaponDefinition> > RepArray;
|
||||||
|
|
||||||
return (bPendingDelete || bDeleteMe || Destroy());
|
replication
|
||||||
}
|
{
|
||||||
|
if (bNetInitial && Role == ROLE_Authority)
|
||||||
public function PrepareSync(
|
LogLevel;
|
||||||
LTI _LTI,
|
}
|
||||||
E_LogLevel _LogLevel,
|
|
||||||
Array<class<KFWeaponDefinition> > _RemoveItems,
|
public simulated function bool SafeDestroy()
|
||||||
bool _ReplaceMode,
|
{
|
||||||
bool _RemoveHRG,
|
`Log_Trace();
|
||||||
bool _RemoveDLC)
|
|
||||||
{
|
return (bPendingDelete || bDeleteMe || Destroy());
|
||||||
`Log_Trace();
|
}
|
||||||
|
|
||||||
LTI = _LTI;
|
public function PrepareSync(LTI _LTI, E_LogLevel _LogLevel)
|
||||||
LogLevel = _LogLevel;
|
{
|
||||||
RemoveItems = _RemoveItems;
|
`Log_Trace();
|
||||||
ReplaceMode = _ReplaceMode;
|
|
||||||
RemoveHRG = _RemoveHRG;
|
LTI = _LTI;
|
||||||
RemoveDLC = _RemoveDLC;
|
LogLevel = _LogLevel;
|
||||||
SyncSize = RemoveItems.Length;
|
}
|
||||||
}
|
|
||||||
|
public function Replicate(const out Array<class<KFWeaponDefinition> > WeapDefs)
|
||||||
private simulated function KFPlayerController GetKFPC()
|
{
|
||||||
{
|
`Log_Trace();
|
||||||
`Log_Trace();
|
|
||||||
|
RepArray = WeapDefs;
|
||||||
if (KFPC != None) return KFPC;
|
RepData.Size = RepArray.Length;
|
||||||
|
|
||||||
KFPC = KFPlayerController(Owner);
|
if (WorldInfo.NetMode == NM_StandAlone)
|
||||||
|
{
|
||||||
if (KFPC == None && ROLE < ROLE_Authority)
|
Progress(RepArray.Length, RepArray.Length);
|
||||||
{
|
return;
|
||||||
KFPC = KFPlayerController(GetALocalPlayerController());
|
}
|
||||||
}
|
|
||||||
|
Sync();
|
||||||
return KFPC;
|
}
|
||||||
}
|
|
||||||
|
private reliable server function Sync()
|
||||||
private simulated function SetPartyInGameWidget()
|
{
|
||||||
{
|
local int LocalIndex;
|
||||||
`Log_Trace();
|
local int GlobalIndex;
|
||||||
|
|
||||||
if (GetKFPC() == None) return;
|
`Log_Trace();
|
||||||
|
|
||||||
if (KFPC.MyGFxManager == None) return;
|
LocalIndex = 0;
|
||||||
if (KFPC.MyGFxManager.PartyWidget == None) return;
|
GlobalIndex = RepData.Transfered;
|
||||||
|
|
||||||
PartyInGameWidget = KFGFxWidget_PartyInGame(KFPC.MyGFxManager.PartyWidget);
|
while (LocalIndex < CAPACITY && GlobalIndex < RepData.Size)
|
||||||
Notification = PartyInGameWidget.Notification;
|
{
|
||||||
}
|
RepData.Items[LocalIndex++] = RepArray[GlobalIndex++];
|
||||||
|
}
|
||||||
private simulated function bool CheckPartyInGameWidget()
|
|
||||||
{
|
if (RepData.Transfered == GlobalIndex) return; // Finished
|
||||||
`Log_Trace();
|
|
||||||
|
RepData.Transfered = GlobalIndex;
|
||||||
if (PartyInGameWidget == None)
|
RepData.Length = LocalIndex;
|
||||||
{
|
|
||||||
SetPartyInGameWidget();
|
Send(RepData);
|
||||||
}
|
|
||||||
|
Progress(RepData.Transfered, RepData.Size);
|
||||||
return (PartyInGameWidget != None);
|
}
|
||||||
}
|
|
||||||
|
private reliable client function Send(ReplicationStruct RD)
|
||||||
private simulated function HideReadyButton()
|
{
|
||||||
{
|
local int LocalIndex;
|
||||||
`Log_Trace();
|
|
||||||
|
`Log_Trace();
|
||||||
if (CheckPartyInGameWidget())
|
|
||||||
{
|
for (LocalIndex = 0; LocalIndex < RD.Length; LocalIndex++)
|
||||||
PartyInGameWidget.SetReadyButtonVisibility(false);
|
{
|
||||||
}
|
RepArray.AddItem(RD.Items[LocalIndex]);
|
||||||
}
|
}
|
||||||
|
|
||||||
private simulated function ShowReadyButton()
|
Progress(RD.Transfered, RD.Size);
|
||||||
{
|
|
||||||
`Log_Trace();
|
Sync();
|
||||||
|
}
|
||||||
if (CheckPartyInGameWidget())
|
|
||||||
{
|
private simulated function Progress(int Value, int Size)
|
||||||
Notification.SetVisible(false);
|
{
|
||||||
PartyInGameWidget.SetReadyButtonVisibility(true);
|
`Log_Trace();
|
||||||
PartyInGameWidget.UpdateReadyButtonText();
|
|
||||||
PartyInGameWidget.UpdateReadyButtonVisibility();
|
`Log_Debug("Replicated:" @ Value @ "/" @ Size);
|
||||||
}
|
|
||||||
}
|
if (ROLE < ROLE_Authority)
|
||||||
|
{
|
||||||
private simulated function UpdateNotification(String Title, String Left, String Right, int Percent)
|
NotifyProgress(Value, Size);
|
||||||
{
|
if (Value >= Size) Finished();
|
||||||
`Log_Trace();
|
}
|
||||||
|
}
|
||||||
if (CheckPartyInGameWidget() && Notification != None)
|
|
||||||
{
|
private simulated function Finished()
|
||||||
Notification.SetString("itemName", Title);
|
{
|
||||||
Notification.SetFloat("percent", Percent);
|
local KFGameReplicationInfo KFGRI;
|
||||||
Notification.SetInt("queue", 0);
|
|
||||||
Notification.SetString("downLoading", Left);
|
`Log_Trace();
|
||||||
Notification.SetString("remaining", Right);
|
|
||||||
Notification.SetObject("notificationInfo", Notification);
|
if (GetGRI(WaitingGRI > WaitingGRIThreshold) == None && WaitingGRI++ < WaitingGRILimit)
|
||||||
Notification.SetVisible(true);
|
{
|
||||||
}
|
`Log_Debug("Finished: Waiting GRI" @ WaitingGRI);
|
||||||
}
|
NotifyWaitingGRI();
|
||||||
|
SetTimer(1.0f, false, nameof(Finished));
|
||||||
private reliable client function ClientSync(class<KFWeaponDefinition> WeapDef)
|
return;
|
||||||
{
|
}
|
||||||
`Log_Trace();
|
|
||||||
|
KFGRI = KFGameReplicationInfo(GRI);
|
||||||
if (WeapDef == None)
|
if (KFGRI != None)
|
||||||
{
|
{
|
||||||
`Log_Fatal("WeapDef is:" @ WeapDef);
|
`Log_Debug("Finished: Trader.static.OverwriteTraderItems");
|
||||||
Cleanup();
|
Trader.static.OverwriteTraderItems(KFGRI, RepArray, LogLevel);
|
||||||
ConsoleCommand("Disconnect");
|
`Log_Info("Trader items successfully synchronized!");
|
||||||
SafeDestroy();
|
}
|
||||||
return;
|
else
|
||||||
}
|
{
|
||||||
|
`Log_Error("Incompatible Game Replication info:" @ String(GRI));
|
||||||
if (!IsTimerActive(nameof(KeepNotification)))
|
if (GRI == None)
|
||||||
{
|
{
|
||||||
SetTimer(0.1f, true, nameof(KeepNotification));
|
NotifyNoneGRI();
|
||||||
}
|
}
|
||||||
|
else
|
||||||
RemoveItems.AddItem(WeapDef);
|
{
|
||||||
|
NotifyIncompatibleGRI();
|
||||||
Recieved = RemoveItems.Length;
|
}
|
||||||
|
}
|
||||||
NotificationHeaderText = "-" @ WeapDef.static.GetItemName();
|
|
||||||
NotificationLeftText = LocalMessage.static.GetLocalizedString(LogLevel, LTI_SyncItems);
|
ShowReadyButton();
|
||||||
NotificationRightText = Recieved @ "/" @ SyncSize;
|
ClientCleanup();
|
||||||
if (SyncSize != 0)
|
}
|
||||||
{
|
|
||||||
NotificationPercent = (float(Recieved) / float(SyncSize)) * 100;
|
private simulated function GameReplicationInfo GetGRI(optional bool ForcedSearch = false)
|
||||||
}
|
{
|
||||||
|
`Log_Trace();
|
||||||
`Log_Debug("ClientSync: -" @ String(WeapDef) @ NotificationRightText);
|
|
||||||
|
if (GRI == None)
|
||||||
ServerSync();
|
{
|
||||||
}
|
GRI = WorldInfo.GRI;
|
||||||
|
}
|
||||||
private simulated function KeepNotification()
|
|
||||||
{
|
if (GRI == None && ForcedSearch)
|
||||||
HideReadyButton();
|
{
|
||||||
UpdateNotification(
|
foreach WorldInfo.DynamicActors(class'GameReplicationInfo', GRI) break;
|
||||||
NotificationHeaderText,
|
}
|
||||||
NotificationLeftText,
|
|
||||||
NotificationRightText,
|
if (WorldInfo.GRI == None && GRI != None)
|
||||||
NotificationPercent);
|
{
|
||||||
}
|
`Log_Warn("Force initialization of WorldInfo.GRI" @ String(GRI));
|
||||||
|
WorldInfo.GRI = GRI;
|
||||||
private simulated reliable client function ClientSyncFinished()
|
}
|
||||||
{
|
|
||||||
local KFGameReplicationInfo KFGRI;
|
return GRI;
|
||||||
|
}
|
||||||
`Log_Trace();
|
|
||||||
|
private simulated function KFPlayerController GetKFPC()
|
||||||
NotificationLeftText = "";
|
{
|
||||||
NotificationRightText = "";
|
`Log_Trace();
|
||||||
NotificationPercent = 0;
|
|
||||||
|
if (KFPC != None) return KFPC;
|
||||||
if (WorldInfo.GRI == None)
|
|
||||||
{
|
KFPC = KFPlayerController(Owner);
|
||||||
`Log_Debug("ClientSyncFinished: Waiting GRI");
|
|
||||||
NotificationHeaderText = LocalMessage.static.GetLocalizedString(LogLevel, LTI_WaitingGRI);
|
if (KFPC == None && ROLE < ROLE_Authority)
|
||||||
NotificationLeftText = String(++WaitingGRI) $ LocalMessage.static.GetLocalizedString(LogLevel, LTI_SecondsShort);
|
{
|
||||||
NotificationRightText = "";
|
KFPC = KFPlayerController(GetALocalPlayerController());
|
||||||
SetTimer(1.0f, false, nameof(ClientSyncFinished));
|
}
|
||||||
return;
|
|
||||||
}
|
return KFPC;
|
||||||
|
}
|
||||||
KFGRI = KFGameReplicationInfo(WorldInfo.GRI);
|
|
||||||
if (KFGRI == None)
|
public reliable client function WriteToChatLocalized(
|
||||||
{
|
E_LTI_LocalMessageType LMT,
|
||||||
`Log_Fatal("Incompatible Replication info:" @ String(WorldInfo.GRI));
|
optional String HexColor,
|
||||||
ClearTimer(nameof(KeepNotification));
|
optional String String1,
|
||||||
UpdateNotification(
|
optional String String2,
|
||||||
LocalMessage.static.GetLocalizedString(LogLevel, LTI_IncompatibleGRI) @ String(WorldInfo.GRI),
|
optional String String3)
|
||||||
LocalMessage.static.GetLocalizedString(LogLevel, LTI_Disconnect), "", 0);
|
{
|
||||||
Cleanup();
|
`Log_Trace();
|
||||||
ConsoleCommand("Disconnect");
|
|
||||||
SafeDestroy();
|
WriteToChat(LocalMessage.static.GetLocalizedString(LogLevel, LMT, String1, String2, String3), HexColor);
|
||||||
return;
|
}
|
||||||
}
|
|
||||||
|
public reliable client function WriteToChat(String Message, optional String HexColor)
|
||||||
NotificationHeaderText = LocalMessage.static.GetLocalizedString(LogLevel, LTI_SyncFinished);
|
{
|
||||||
NotificationLeftText = "";
|
local KFGFxHudWrapper HUD;
|
||||||
NotificationRightText = "";
|
|
||||||
NotificationPercent = 0;
|
`Log_Trace();
|
||||||
|
|
||||||
Trader.static.ModifyTrader(KFGRI, RemoveItems, ReplaceMode, RemoveHRG, RemoveDLC, LogLevel);
|
if (GetKFPC() == None) return;
|
||||||
`Log_Debug("ClientSyncFinished: Trader.static.ModifyTrader");
|
|
||||||
|
if (KFPC.MyGFxManager.PartyWidget != None && KFPC.MyGFxManager.PartyWidget.PartyChatWidget != None)
|
||||||
ClearTimer(nameof(KeepNotification));
|
{
|
||||||
ShowReadyButton();
|
KFPC.MyGFxManager.PartyWidget.PartyChatWidget.SetVisible(true);
|
||||||
|
KFPC.MyGFxManager.PartyWidget.PartyChatWidget.AddChatMessage(Message, HexColor);
|
||||||
Cleanup();
|
}
|
||||||
|
|
||||||
SafeDestroy();
|
HUD = KFGFxHudWrapper(KFPC.myHUD);
|
||||||
}
|
if (HUD != None && HUD.HUDMovie != None && HUD.HUDMovie.HudChatBox != None)
|
||||||
|
{
|
||||||
private reliable server function Cleanup()
|
HUD.HUDMovie.HudChatBox.AddChatMessage(Message, HexColor);
|
||||||
{
|
}
|
||||||
`Log_Trace();
|
}
|
||||||
|
|
||||||
`Log_Debug("Cleanup");
|
private simulated function SetPartyInGameWidget()
|
||||||
if (!LTI.DestroyRepInfo(Controller(Owner)))
|
{
|
||||||
{
|
`Log_Trace();
|
||||||
`Log_Debug("Cleanup (forced)");
|
|
||||||
SafeDestroy();
|
if (GetKFPC() == None) return;
|
||||||
}
|
|
||||||
}
|
if (KFPC.MyGFxManager == None) return;
|
||||||
|
if (KFPC.MyGFxManager.PartyWidget == None) return;
|
||||||
public reliable server function ServerSync()
|
|
||||||
{
|
PartyInGameWidget = KFGFxWidget_PartyInGame(KFPC.MyGFxManager.PartyWidget);
|
||||||
`Log_Trace();
|
Notification = PartyInGameWidget.Notification;
|
||||||
|
}
|
||||||
PendingSync = false;
|
|
||||||
|
private simulated function bool CheckPartyInGameWidget()
|
||||||
if (bPendingDelete || bDeleteMe) return;
|
{
|
||||||
|
`Log_Trace();
|
||||||
if (SyncSize <= Recieved || WorldInfo.NetMode == NM_StandAlone)
|
|
||||||
{
|
if (PartyInGameWidget == None)
|
||||||
`Log_Debug("ServerSync: Finished");
|
{
|
||||||
ClientSyncFinished();
|
SetPartyInGameWidget();
|
||||||
}
|
}
|
||||||
else
|
|
||||||
{
|
return (PartyInGameWidget != None);
|
||||||
if (Recieved < RemoveItems.Length)
|
}
|
||||||
{
|
|
||||||
`Log_Debug("ServerSync[-]:" @ (Recieved + 1) @ "/" @ SyncSize @ RemoveItems[Recieved]);
|
private simulated function HideReadyButton()
|
||||||
ClientSync(RemoveItems[Recieved++]);
|
{
|
||||||
}
|
`Log_Trace();
|
||||||
}
|
|
||||||
}
|
if (CheckPartyInGameWidget())
|
||||||
|
{
|
||||||
defaultproperties
|
PartyInGameWidget.SetReadyButtonVisibility(false);
|
||||||
{
|
}
|
||||||
bAlwaysRelevant = false
|
}
|
||||||
bOnlyRelevantToOwner = true
|
|
||||||
bSkipActorPropertyReplication = false
|
private simulated function ShowReadyButton()
|
||||||
|
{
|
||||||
PendingSync = false
|
`Log_Trace();
|
||||||
Recieved = 0
|
|
||||||
|
ClearTimer(nameof(KeepNotification));
|
||||||
NotificationPercent = 0
|
|
||||||
WaitingGRI = 0
|
if (CheckPartyInGameWidget())
|
||||||
}
|
{
|
||||||
|
Notification.SetVisible(false);
|
||||||
|
PartyInGameWidget.SetReadyButtonVisibility(true);
|
||||||
|
PartyInGameWidget.UpdateReadyButtonText();
|
||||||
|
PartyInGameWidget.UpdateReadyButtonVisibility();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private simulated function UpdateNotification(String Title, String Left, String Right, int Percent)
|
||||||
|
{
|
||||||
|
`Log_Trace();
|
||||||
|
|
||||||
|
if (CheckPartyInGameWidget() && Notification != None)
|
||||||
|
{
|
||||||
|
Notification.SetString("itemName", Title);
|
||||||
|
Notification.SetFloat("percent", Percent);
|
||||||
|
Notification.SetInt("queue", 0);
|
||||||
|
Notification.SetString("downLoading", Left);
|
||||||
|
Notification.SetString("remaining", Right);
|
||||||
|
Notification.SetObject("notificationInfo", Notification);
|
||||||
|
Notification.SetVisible(true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private simulated function KeepNotification()
|
||||||
|
{
|
||||||
|
HideReadyButton();
|
||||||
|
UpdateNotification(
|
||||||
|
NotificationHeaderText,
|
||||||
|
NotificationLeftText,
|
||||||
|
NotificationRightText,
|
||||||
|
NotificationPercent);
|
||||||
|
}
|
||||||
|
|
||||||
|
private simulated function ClientCleanup()
|
||||||
|
{
|
||||||
|
`Log_Debug("Cleanup");
|
||||||
|
ServerCleanup();
|
||||||
|
SafeDestroy();
|
||||||
|
}
|
||||||
|
|
||||||
|
private reliable server function ServerCleanup()
|
||||||
|
{
|
||||||
|
`Log_Trace();
|
||||||
|
|
||||||
|
`Log_Debug("Cleanup");
|
||||||
|
if (!LTI.DestroyRepInfo(GetKFPC()))
|
||||||
|
{
|
||||||
|
`Log_Debug("Cleanup (forced)");
|
||||||
|
SafeDestroy();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private simulated function NotifyWaitingGRI()
|
||||||
|
{
|
||||||
|
if (!IsTimerActive(nameof(KeepNotification)))
|
||||||
|
{
|
||||||
|
SetTimer(0.1f, true, nameof(KeepNotification));
|
||||||
|
}
|
||||||
|
|
||||||
|
NotificationHeaderText = LocalMessage.static.GetLocalizedString(LogLevel, LTI_WaitingGRI);
|
||||||
|
NotificationLeftText = String(WaitingGRI) $ LocalMessage.static.GetLocalizedString(LogLevel, LTI_SecondsShort);
|
||||||
|
NotificationRightText = LocalMessage.static.GetLocalizedString(LogLevel, LTI_PleaseWait);
|
||||||
|
NotificationPercent = 0;
|
||||||
|
KeepNotification();
|
||||||
|
}
|
||||||
|
|
||||||
|
private simulated function NotifyProgress(int Value, int Size)
|
||||||
|
{
|
||||||
|
if (!IsTimerActive(nameof(KeepNotification)))
|
||||||
|
{
|
||||||
|
SetTimer(0.1f, true, nameof(KeepNotification));
|
||||||
|
}
|
||||||
|
|
||||||
|
NotificationHeaderText = LocalMessage.static.GetLocalizedString(LogLevel, LTI_SyncItems);
|
||||||
|
NotificationLeftText = Value @ "/" @ Size;
|
||||||
|
NotificationRightText = LocalMessage.static.GetLocalizedString(LogLevel, LTI_PleaseWait);
|
||||||
|
NotificationPercent = (float(Value) / float(Size)) * 100;
|
||||||
|
KeepNotification();
|
||||||
|
}
|
||||||
|
|
||||||
|
private simulated function NotifyIncompatibleGRI()
|
||||||
|
{
|
||||||
|
WriteToChatLocalized(
|
||||||
|
LTI_IncompatibleGRI,
|
||||||
|
class'KFLocalMessage'.default.InteractionColor,
|
||||||
|
String(GRI.class));
|
||||||
|
WriteToChatLocalized(
|
||||||
|
LTI_IncompatibleGRIWarning,
|
||||||
|
class'KFLocalMessage'.default.InteractionColor);
|
||||||
|
}
|
||||||
|
|
||||||
|
private simulated function NotifyNoneGRI()
|
||||||
|
{
|
||||||
|
WriteToChatLocalized(
|
||||||
|
LTI_NoneGRI,
|
||||||
|
class'KFLocalMessage'.default.InteractionColor);
|
||||||
|
WriteToChatLocalized(
|
||||||
|
LTI_NoneGRIWarning,
|
||||||
|
class'KFLocalMessage'.default.InteractionColor);
|
||||||
|
}
|
||||||
|
|
||||||
|
defaultproperties
|
||||||
|
{
|
||||||
|
bAlwaysRelevant = false
|
||||||
|
bOnlyRelevantToOwner = true
|
||||||
|
bSkipActorPropertyReplication = false
|
||||||
|
|
||||||
|
PendingSync = false
|
||||||
|
|
||||||
|
NotificationPercent = 0
|
||||||
|
WaitingGRI = 0
|
||||||
|
WaitingGRIThreshold = 10
|
||||||
|
WaitingGRILimit = 20
|
||||||
|
}
|
||||||
|
60
LTI/Classes/Mut.uc
Normal file
60
LTI/Classes/Mut.uc
Normal file
@ -0,0 +1,60 @@
|
|||||||
|
class Mut extends KFMutator;
|
||||||
|
|
||||||
|
var private LTI LTI;
|
||||||
|
|
||||||
|
public simulated function bool SafeDestroy()
|
||||||
|
{
|
||||||
|
return (bPendingDelete || bDeleteMe || Destroy());
|
||||||
|
}
|
||||||
|
|
||||||
|
public event PreBeginPlay()
|
||||||
|
{
|
||||||
|
Super.PreBeginPlay();
|
||||||
|
|
||||||
|
if (WorldInfo.NetMode == NM_Client) return;
|
||||||
|
|
||||||
|
foreach WorldInfo.DynamicActors(class'LTI', LTI)
|
||||||
|
{
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (LTI == None)
|
||||||
|
{
|
||||||
|
LTI = WorldInfo.Spawn(class'LTI');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (LTI == None)
|
||||||
|
{
|
||||||
|
`Log_Base("FATAL: Can't Spawn 'LTI'");
|
||||||
|
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)
|
||||||
|
{
|
||||||
|
LTI.NotifyLogin(C);
|
||||||
|
|
||||||
|
Super.NotifyLogin(C);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function NotifyLogout(Controller C)
|
||||||
|
{
|
||||||
|
LTI.NotifyLogout(C);
|
||||||
|
|
||||||
|
Super.NotifyLogout(C);
|
||||||
|
}
|
||||||
|
|
||||||
|
DefaultProperties
|
||||||
|
{
|
||||||
|
GroupNames.Add("TraderItems")
|
||||||
|
}
|
@ -1,43 +1,43 @@
|
|||||||
class OfficialWeapons extends Object
|
class OfficialWeapons extends Object
|
||||||
config(LTI);
|
config(LTI);
|
||||||
|
|
||||||
const Trader = class'Trader';
|
const Trader = class'Trader';
|
||||||
const DefaultComment = "Auto-generated list of official weapons for your convenience, copy-paste ready";
|
const DefaultComment = "Auto-generated list of official weapons for your convenience, copy-paste ready";
|
||||||
|
|
||||||
var private config String Comment;
|
var private config String Comment;
|
||||||
var private config Array<String> Item;
|
var private config Array<String> Item;
|
||||||
|
|
||||||
private delegate int ByName(String A, String B)
|
private delegate int ByName(String A, String B)
|
||||||
{
|
{
|
||||||
return A > B ? -1 : 0;
|
return A > B ? -1 : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function Update(bool Enabled)
|
public static function Update(bool Enabled)
|
||||||
{
|
{
|
||||||
local Array<class<KFWeaponDefinition> > KFWeapDefs;
|
local Array<class<KFWeaponDefinition> > KFWeapDefs;
|
||||||
local class<KFWeaponDefinition> KFWeapDef;
|
local class<KFWeaponDefinition> KFWeapDef;
|
||||||
|
|
||||||
if (!Enabled) return;
|
if (!Enabled) return;
|
||||||
|
|
||||||
KFWeapDefs = Trader.static.GetTraderWeapDefs();
|
KFWeapDefs = Trader.static.GetTraderWeapDefs();
|
||||||
|
|
||||||
if (default.Item.Length != KFWeapDefs.Length || default.Comment != DefaultComment)
|
if (default.Item.Length != KFWeapDefs.Length || default.Comment != DefaultComment)
|
||||||
{
|
{
|
||||||
default.Comment = DefaultComment;
|
default.Comment = DefaultComment;
|
||||||
default.Item.Length = 0;
|
default.Item.Length = 0;
|
||||||
|
|
||||||
foreach KFWeapDefs(KFWeapDef)
|
foreach KFWeapDefs(KFWeapDef)
|
||||||
{
|
{
|
||||||
default.Item.AddItem(KFWeapDef.GetPackageName() $ "." $ KFWeapDef);
|
default.Item.AddItem(KFWeapDef.GetPackageName() $ "." $ KFWeapDef);
|
||||||
}
|
}
|
||||||
|
|
||||||
default.Item.Sort(ByName);
|
default.Item.Sort(ByName);
|
||||||
|
|
||||||
StaticSaveConfig();
|
StaticSaveConfig();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
defaultproperties
|
defaultproperties
|
||||||
{
|
{
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -1,105 +1,105 @@
|
|||||||
class RemoveItems extends Object
|
class RemoveItems extends Object
|
||||||
dependson(LTI)
|
dependson(LTI)
|
||||||
config(LTI);
|
config(LTI);
|
||||||
|
|
||||||
var public config bool bAll;
|
var public config bool bAll;
|
||||||
var public config bool bHRG;
|
var public config bool bHRG;
|
||||||
var public config bool bDLC;
|
var public config bool bDLC;
|
||||||
var private config Array<String> Item;
|
var private config Array<String> Item;
|
||||||
|
|
||||||
public static function InitConfig(int Version, int LatestVersion)
|
public static function InitConfig(int Version, int LatestVersion)
|
||||||
{
|
{
|
||||||
switch (Version)
|
switch (Version)
|
||||||
{
|
{
|
||||||
case `NO_CONFIG:
|
case `NO_CONFIG:
|
||||||
ApplyDefault();
|
ApplyDefault();
|
||||||
|
|
||||||
case 1:
|
case 1:
|
||||||
default.bHRG = false;
|
default.bHRG = false;
|
||||||
default.bDLC = false;
|
default.bDLC = false;
|
||||||
|
|
||||||
default: break;
|
default: break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (LatestVersion != Version)
|
if (LatestVersion != Version)
|
||||||
{
|
{
|
||||||
StaticSaveConfig();
|
StaticSaveConfig();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private static function ApplyDefault()
|
private static function ApplyDefault()
|
||||||
{
|
{
|
||||||
default.bAll = false;
|
default.bAll = false;
|
||||||
default.bHRG = false;
|
default.bHRG = false;
|
||||||
default.bDLC = false;
|
default.bDLC = false;
|
||||||
default.Item.Length = 0;
|
default.Item.Length = 0;
|
||||||
default.Item.AddItem("KFGame.KFWeapDef_9mmDual");
|
default.Item.AddItem("KFGame.KFWeapDef_9mmDual");
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function Array<class<KFWeaponDefinition> > Load(E_LogLevel LogLevel)
|
public static function Array<class<KFWeaponDefinition> > Load(E_LogLevel LogLevel)
|
||||||
{
|
{
|
||||||
local Array<class<KFWeaponDefinition> > ItemList;
|
local Array<class<KFWeaponDefinition> > ItemList;
|
||||||
local class<KFWeaponDefinition> ItemWeapDef;
|
local class<KFWeaponDefinition> ItemWeapDef;
|
||||||
local class<KFWeapon> ItemWeapon;
|
local class<KFWeapon> ItemWeapon;
|
||||||
local String ItemRaw;
|
local String ItemRaw;
|
||||||
local int Line;
|
local int Line;
|
||||||
|
|
||||||
`Log_Info("Load items to remove:");
|
`Log_Info("Load items to remove:");
|
||||||
if (default.bAll)
|
if (default.bAll)
|
||||||
{
|
{
|
||||||
`Log_Info("Remove all default items");
|
`Log_Info("Remove all default items");
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if (default.bHRG)
|
if (default.bHRG)
|
||||||
{
|
{
|
||||||
`Log_Info("Remove all HRG items");
|
`Log_Info("Remove all HRG items");
|
||||||
}
|
}
|
||||||
if (default.bDLC)
|
if (default.bDLC)
|
||||||
{
|
{
|
||||||
`Log_Info("Remove all DLC items");
|
`Log_Info("Remove all DLC items");
|
||||||
}
|
}
|
||||||
|
|
||||||
foreach default.Item(ItemRaw, Line)
|
foreach default.Item(ItemRaw, Line)
|
||||||
{
|
{
|
||||||
ItemWeapDef = class<KFWeaponDefinition>(DynamicLoadObject(ItemRaw, class'Class'));
|
ItemWeapDef = class<KFWeaponDefinition>(DynamicLoadObject(ItemRaw, class'Class'));
|
||||||
if (ItemWeapDef == None)
|
if (ItemWeapDef == None)
|
||||||
{
|
{
|
||||||
`Log_Warn("[" $ Line + 1 $ "]" @ "Can't load weapon definition:" @ ItemRaw);
|
`Log_Warn("[" $ Line + 1 $ "]" @ "Can't load weapon definition:" @ ItemRaw);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
ItemWeapon = class<KFWeapon>(DynamicLoadObject(ItemWeapDef.default.WeaponClassPath, class'Class'));
|
ItemWeapon = class<KFWeapon>(DynamicLoadObject(ItemWeapDef.default.WeaponClassPath, class'Class'));
|
||||||
if (ItemWeapon == None)
|
if (ItemWeapon == None)
|
||||||
{
|
{
|
||||||
`Log_Warn("[" $ Line + 1 $ "]" @ "Can't load weapon:" @ ItemWeapDef.default.WeaponClassPath);
|
`Log_Warn("[" $ Line + 1 $ "]" @ "Can't load weapon:" @ ItemWeapDef.default.WeaponClassPath);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ItemList.Find(ItemWeapDef) != INDEX_NONE)
|
if (ItemList.Find(ItemWeapDef) != INDEX_NONE)
|
||||||
{
|
{
|
||||||
`Log_Warn("[" $ Line + 1 $ "]" @ "Duplicate item:" @ ItemRaw @ "(skip)");
|
`Log_Warn("[" $ Line + 1 $ "]" @ "Duplicate item:" @ ItemRaw @ "(skip)");
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
ItemList.AddItem(ItemWeapDef);
|
ItemList.AddItem(ItemWeapDef);
|
||||||
`Log_Debug("[" $ Line + 1 $ "]" @ "Loaded successfully:" @ ItemRaw);
|
`Log_Debug("[" $ Line + 1 $ "]" @ "Loaded successfully:" @ ItemRaw);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ItemList.Length == default.Item.Length)
|
if (ItemList.Length == default.Item.Length)
|
||||||
{
|
{
|
||||||
`Log_Info("Items to remove list loaded successfully (" $ ItemList.Length @ "entries)");
|
`Log_Info("Items to remove list loaded successfully (" $ ItemList.Length @ "entries)");
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
`Log_Info("Items to remove list: loaded" @ ItemList.Length @ "of" @ default.Item.Length @ "entries");
|
`Log_Info("Items to remove list: loaded" @ ItemList.Length @ "of" @ default.Item.Length @ "entries");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return ItemList;
|
return ItemList;
|
||||||
}
|
}
|
||||||
|
|
||||||
defaultproperties
|
defaultproperties
|
||||||
{
|
{
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -1,119 +1,137 @@
|
|||||||
class Trader extends Object
|
class Trader extends Object
|
||||||
abstract;
|
abstract;
|
||||||
|
|
||||||
private delegate int ByPrice(class<KFWeaponDefinition> A, class<KFWeaponDefinition> B)
|
private delegate int ByPrice(class<KFWeaponDefinition> A, class<KFWeaponDefinition> B)
|
||||||
{
|
{
|
||||||
return A.default.BuyPrice > B.default.BuyPrice ? -1 : 0;
|
return A.default.BuyPrice > B.default.BuyPrice ? -1 : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function KFGFxObject_TraderItems GetTraderItems(optional KFGameReplicationInfo KFGRI = None, optional E_LogLevel LogLevel = LL_Trace)
|
public static function KFGFxObject_TraderItems GetTraderItems(optional KFGameReplicationInfo KFGRI = None, optional E_LogLevel LogLevel = LL_Trace)
|
||||||
{
|
{
|
||||||
local String TraderItemsPath;
|
local String TraderItemsPath;
|
||||||
|
|
||||||
if (KFGRI == None)
|
if (KFGRI == None)
|
||||||
{
|
{
|
||||||
TraderItemsPath = class'KFGameReplicationInfo'.default.TraderItemsPath;
|
TraderItemsPath = class'KFGameReplicationInfo'.default.TraderItemsPath;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
TraderItemsPath = KFGRI.TraderItemsPath;
|
TraderItemsPath = KFGRI.TraderItemsPath;
|
||||||
}
|
}
|
||||||
|
|
||||||
return KFGFxObject_TraderItems(DynamicLoadObject(TraderItemsPath, class'KFGFxObject_TraderItems'));
|
return KFGFxObject_TraderItems(DynamicLoadObject(TraderItemsPath, class'KFGFxObject_TraderItems'));
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function Array<class<KFWeaponDefinition> > GetTraderWeapDefs(optional KFGameReplicationInfo KFGRI = None,optional E_LogLevel LogLevel = LL_Trace)
|
public static function Array<class<KFWeaponDefinition> > GetTraderWeapDefs(optional KFGameReplicationInfo KFGRI = None,optional E_LogLevel LogLevel = LL_Trace)
|
||||||
{
|
{
|
||||||
local Array<class<KFWeaponDefinition> > KFWeapDefs;
|
local Array<class<KFWeaponDefinition> > KFWeapDefs;
|
||||||
local KFGFxObject_TraderItems TraderItems;
|
local KFGFxObject_TraderItems TraderItems;
|
||||||
local STraderItem Item;
|
local STraderItem Item;
|
||||||
|
|
||||||
TraderItems = GetTraderItems(KFGRI, LogLevel);
|
TraderItems = GetTraderItems(KFGRI, LogLevel);
|
||||||
|
|
||||||
foreach TraderItems.SaleItems(Item)
|
foreach TraderItems.SaleItems(Item)
|
||||||
{
|
{
|
||||||
if (Item.WeaponDef != None)
|
if (Item.WeaponDef != None)
|
||||||
{
|
{
|
||||||
KFWeapDefs.AddItem(Item.WeaponDef);
|
KFWeapDefs.AddItem(Item.WeaponDef);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return KFWeapDefs;
|
return KFWeapDefs;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function Array<class<KFWeapon> > GetTraderWeapons(optional KFGameReplicationInfo KFGRI = None,optional E_LogLevel LogLevel = LL_Trace)
|
public static function Array<class<KFWeapon> > GetTraderWeapons(optional KFGameReplicationInfo KFGRI = None,optional E_LogLevel LogLevel = LL_Trace)
|
||||||
{
|
{
|
||||||
local Array<class<KFWeapon> > KFWeapons;
|
local Array<class<KFWeapon> > KFWeapons;
|
||||||
local class<KFWeapon> KFWeapon;
|
local class<KFWeapon> KFWeapon;
|
||||||
local KFGFxObject_TraderItems TraderItems;
|
local KFGFxObject_TraderItems TraderItems;
|
||||||
local STraderItem Item;
|
local STraderItem Item;
|
||||||
|
|
||||||
TraderItems = GetTraderItems(KFGRI, LogLevel);
|
TraderItems = GetTraderItems(KFGRI, LogLevel);
|
||||||
|
|
||||||
foreach TraderItems.SaleItems(Item)
|
foreach TraderItems.SaleItems(Item)
|
||||||
{
|
{
|
||||||
if (Item.WeaponDef != None)
|
if (Item.WeaponDef != None)
|
||||||
{
|
{
|
||||||
KFWeapon = class<KFWeapon> (DynamicLoadObject(Item.WeaponDef.default.WeaponClassPath, class'Class'));
|
KFWeapon = class<KFWeapon> (DynamicLoadObject(Item.WeaponDef.default.WeaponClassPath, class'Class'));
|
||||||
if (KFWeapon != None)
|
if (KFWeapon != None)
|
||||||
{
|
{
|
||||||
KFWeapons.AddItem(KFWeapon);
|
KFWeapons.AddItem(KFWeapon);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return KFWeapons;
|
return KFWeapons;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static simulated function ModifyTrader(
|
public static simulated function ModifyTrader(
|
||||||
KFGameReplicationInfo KFGRI,
|
KFGameReplicationInfo KFGRI,
|
||||||
Array<class<KFWeaponDefinition> > RemoveItems,
|
Array<class<KFWeaponDefinition> > RemoveItems,
|
||||||
bool ReplaceMode,
|
bool ReplaceMode,
|
||||||
bool RemoveHRG,
|
bool RemoveHRG,
|
||||||
bool RemoveDLC,
|
bool RemoveDLC,
|
||||||
E_LogLevel LogLevel)
|
E_LogLevel LogLevel)
|
||||||
{
|
{
|
||||||
local KFGFxObject_TraderItems TraderItems;
|
local KFGFxObject_TraderItems TraderItems;
|
||||||
local STraderItem Item;
|
local STraderItem Item;
|
||||||
local class<KFWeaponDefinition> WeapDef;
|
local Array<class<KFWeaponDefinition> > WeapDefs;
|
||||||
local Array<class<KFWeaponDefinition> > WeapDefs;
|
|
||||||
local int MaxItemID;
|
`Log_TraceStatic();
|
||||||
|
|
||||||
`Log_TraceStatic();
|
TraderItems = GetTraderItems(KFGRI, LogLevel);
|
||||||
|
|
||||||
TraderItems = GetTraderItems(KFGRI, LogLevel);
|
if (!ReplaceMode)
|
||||||
|
{
|
||||||
if (!ReplaceMode)
|
foreach TraderItems.SaleItems(Item)
|
||||||
{
|
{
|
||||||
foreach TraderItems.SaleItems(Item)
|
if (Item.WeaponDef != None
|
||||||
{
|
&& RemoveItems.Find(Item.WeaponDef) == INDEX_NONE
|
||||||
if (Item.WeaponDef != None
|
&& (!RemoveHRG || (RemoveHRG && InStr(Item.WeaponDef, "_HRG", true) == INDEX_NONE))
|
||||||
&& RemoveItems.Find(Item.WeaponDef) == INDEX_NONE
|
&& (!RemoveDLC || (RemoveDLC && Item.WeaponDef.default.SharedUnlockId == SCU_None)))
|
||||||
&& (!RemoveHRG || (RemoveHRG && InStr(Item.WeaponDef, "_HRG", true) == INDEX_NONE))
|
{
|
||||||
&& (!RemoveDLC || (RemoveDLC && Item.WeaponDef.default.SharedUnlockId == SCU_None)))
|
WeapDefs.AddItem(Item.WeaponDef);
|
||||||
{
|
}
|
||||||
WeapDefs.AddItem(Item.WeaponDef);
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
WeapDefs.Sort(ByPrice);
|
||||||
|
|
||||||
WeapDefs.Sort(ByPrice);
|
OverwriteTraderItems(KFGRI, WeapDefs, LogLevel);
|
||||||
|
}
|
||||||
TraderItems.SaleItems.Length = 0;
|
|
||||||
MaxItemID = 0;
|
public static simulated function OverwriteTraderItems(
|
||||||
foreach WeapDefs(WeapDef)
|
KFGameReplicationInfo KFGRI,
|
||||||
{
|
const out Array<class<KFWeaponDefinition> > WeapDefs,
|
||||||
Item.WeaponDef = WeapDef;
|
E_LogLevel LogLevel)
|
||||||
Item.ItemID = ++MaxItemID;
|
{
|
||||||
TraderItems.SaleItems.AddItem(Item);
|
local KFGFxObject_TraderItems TraderItems;
|
||||||
}
|
local STraderItem Item;
|
||||||
|
local class<KFWeaponDefinition> WeapDef;
|
||||||
TraderItems.SetItemsInfo(TraderItems.SaleItems);
|
local int MaxItemID;
|
||||||
|
|
||||||
KFGRI.TraderItems = TraderItems;
|
`Log_TraceStatic();
|
||||||
}
|
|
||||||
|
TraderItems = GetTraderItems(KFGRI, LogLevel);
|
||||||
defaultproperties
|
|
||||||
{
|
TraderItems.SaleItems.Length = 0;
|
||||||
|
MaxItemID = 0;
|
||||||
}
|
|
||||||
|
`Log_Debug("Trader Items:");
|
||||||
|
foreach WeapDefs(WeapDef)
|
||||||
|
{
|
||||||
|
Item.WeaponDef = WeapDef;
|
||||||
|
Item.ItemID = MaxItemID++;
|
||||||
|
TraderItems.SaleItems.AddItem(Item);
|
||||||
|
`Log_Debug("[" $ MaxItemID $ "]" @ String(WeapDef));
|
||||||
|
}
|
||||||
|
|
||||||
|
TraderItems.SetItemsInfo(TraderItems.SaleItems);
|
||||||
|
|
||||||
|
KFGRI.TraderItems = TraderItems;
|
||||||
|
}
|
||||||
|
|
||||||
|
defaultproperties
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
@ -1,20 +1,20 @@
|
|||||||
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_Info,
|
LL_Info,
|
||||||
LL_Debug,
|
LL_Debug,
|
||||||
LL_Trace,
|
LL_Trace,
|
||||||
LL_All
|
LL_All
|
||||||
};
|
};
|
||||||
|
|
||||||
defaultproperties
|
defaultproperties
|
||||||
{
|
{
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -1,2 +1,2 @@
|
|||||||
// Constants
|
// Constants
|
||||||
`define NO_CONFIG 0
|
`define NO_CONFIG 0
|
||||||
|
@ -1,3 +1,3 @@
|
|||||||
// Imports
|
// Imports
|
||||||
`include(Logger.uci)
|
`include(Logger.uci)
|
||||||
`include(Constants.uci)
|
`include(Constants.uci)
|
||||||
|
@ -1,15 +1,15 @@
|
|||||||
// Logger
|
// Logger
|
||||||
`define Log_Tag 'LTI'
|
`define Log_Tag 'LTI'
|
||||||
|
|
||||||
`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_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/LTI.chn
Normal file
BIN
Localization/CHN/LTI.chn
Normal file
Binary file not shown.
BIN
Localization/CHT/LTI.cht
Normal file
BIN
Localization/CHT/LTI.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-LootedTraderInventory[/img] [img]https://img.shields.io/steam/downloads/2864857909[/img] [img]https://img.shields.io/steam/favorites/2864857909[/img] [img]https://img.shields.io/steam/update-date/2864857909[/img] [url=https://steamcommunity.com/sharedfiles/filedetails/changelog/2864857909][img]https://img.shields.io/github/v/tag/GenZmeY/KF2-LootedTraderInventory[/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-LootedTraderInventory[/img] [img]https://img.shields.io/steam/favorites/2864857909[/img] [img]https://img.shields.io/steam/update-date/2864857909[/img] [url=https://steamcommunity.com/sharedfiles/filedetails/changelog/2864857909][img]https://img.shields.io/github/v/tag/GenZmeY/KF2-LootedTraderInventory[/img][/url]
|
||||||
|
|
||||||
[h1]Features[/h1]
|
[h1]Features[/h1]
|
||||||
[list]
|
[list]
|
||||||
@ -19,11 +19,17 @@ https://forums.tripwireinteractive.com/index.php?threads/whitelisting-mods-and-m
|
|||||||
[h1]Usage (single player)[/h1]
|
[h1]Usage (single player)[/h1]
|
||||||
[olist]
|
[olist]
|
||||||
[*]Subscribe to this mutator;
|
[*]Subscribe to this mutator;
|
||||||
[*]Start KF2;
|
[*]Create a file [b](*)[/b]: [b]C:\Users\<username>\Documents\My Games\KillingFloor2\KFGame\Config\KFLTI.ini[/b]
|
||||||
[*]Open console (~) and input:
|
with the following content:
|
||||||
[b]open KF-BioticsLab?Mutator=LTI.LTIMut[/b]
|
[b][LTI.LTI]
|
||||||
(replace the map and add the parameters you need)
|
Version=0[/b]
|
||||||
[*]<Enter>.
|
[*]Start KF2;
|
||||||
|
[*]Open console (~) and start any map with LTI (this will generate the default KFLTI.ini content):
|
||||||
|
[b]open KF-BioticsLab?Mutator=LTI.Mut[/b]
|
||||||
|
[*]Close the game and configure LTI as you need (see the [b]Setup (KFLTI.ini)[/b] section below);
|
||||||
|
[*]Start KF2, open the console, start the game:
|
||||||
|
[b]open KF-BioticsLab?Mutator=LTI.Mut[/b]
|
||||||
|
(replace the map and add the parameters you need).
|
||||||
[/olist]
|
[/olist]
|
||||||
[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]
|
||||||
@ -34,32 +40,33 @@ https://forums.tripwireinteractive.com/index.php?threads/whitelisting-mods-and-m
|
|||||||
❗️ If there are several [b]DownloadManagers=[/b] then the line above should be the first ❗️
|
❗️ If there are several [b]DownloadManagers=[/b] then the line above should be the first ❗️
|
||||||
[*]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=2864857909[/b]
|
[b]ServerSubscribedWorkshopItems=2864857909[/b]
|
||||||
[*]Start the server and wait until the mutator is downloading;
|
[*]Start the server and wait while the mutator is downloading;
|
||||||
[*]Add mutator to server start parameters: [b]?Mutator=LTI.LTIMut[/b] and restart the server.
|
[*]When the download is complete, close the server;
|
||||||
[/olist]
|
[*]Create a file [b](*)[/b]: [b]<kf2-server>\KFGame\Config\KFLTI.ini[/b]
|
||||||
|
with the following content:
|
||||||
[h1]Important setup information[/h1]
|
|
||||||
The config should be created on first start, but now the game contains a bug that initializes the config values randomly if they are not explicitly set. Thus, the config may have incorrect values or not be created at all.
|
|
||||||
So if you are using this mutator for the first time, I highly recommend doing the following:
|
|
||||||
[olist]
|
|
||||||
[*]Create (modify) [b]KFLTI.ini[/b] manually. Put the following content there:
|
|
||||||
[b][LTI.LTI]
|
[b][LTI.LTI]
|
||||||
Version=0[/b]
|
Version=0[/b]
|
||||||
[*]Start the game/server with LTI to generate the contents of the config
|
[*]Add mutator to server start parameters: [b]?Mutator=LTI.Mut[/b] and start the server (this will generate the default KFLTI.ini content);
|
||||||
[*]Close the game/server
|
[*]Close the server and configure LTI as you need (see the [b]Setup (KFLTI.ini)[/b] section below);
|
||||||
|
[*]Start the server (with [b]?Mutator=LTI.Mut[/b]) again.
|
||||||
[/olist]
|
[/olist]
|
||||||
[b]Right now this is the only way to correctly create the default config.[/b]
|
|
||||||
|
[h1][b](*)[/b] Buggy config variables initialization[/h1]
|
||||||
|
LTI, like many other mutators, initializes the config by relying on the unreal script feature which uses default values for each data type that is not explicitly specified. For the int type (which is used to store the config version) this is zero - detecting zero allows to understand that the mutator is being used for the first time (which means it's need to generate a config). But now the game contains a bug that initializes the config values randomly if they are not explicitly set. Thus, the config may have incorrect values or not be created at all. This is why I recommend explicitly set [b]Version=0[/b] in the config for the first time.
|
||||||
|
|
||||||
Unfortunately I can't do anything about it because it's a game problem (not mutator). I hope TWI fixes this someday.
|
Unfortunately I can't do anything about it because it's a game problem (not mutator). I hope TWI fixes this someday.
|
||||||
|
|
||||||
[h1]Setup (KFLTI.ini)[/h1]
|
[h1]Setup (KFLTI.ini)[/h1]
|
||||||
[list]
|
[list]
|
||||||
[*]Set [b]bOfficialWeaponsList=True[/b] to have an auto-updated list of all official weapons in the config (for a convenient copy-paste).
|
[*]Set [b]bOfficialWeaponsList=True[/b] to have an auto-updated list of all official weapons in the config (for a convenient copy-paste) or leave it [b]False[/b] if you want a clean config without unnecessary things.
|
||||||
[*]Use [b][LTI.RemoveItems][/b] to remove items from the trader inventory.
|
[*]Use [b][LTI.RemoveItems][/b] to remove items from the trader inventory.
|
||||||
example: [b]Item=KFGame.KFWeapDef_Mac10[/b] will remove MAC10 from sale.
|
example: [b]Item=KFGame.KFWeapDef_Mac10[/b] will remove MAC10 from sale.
|
||||||
|
|
||||||
[*]Set [b]bHRG=True[/b] to remove HRG items.
|
[*]Set [b]bHRG=True[/b] to remove HRG items.
|
||||||
[*]Set [b]bDLC=True[/b] to remove DLC items.
|
[*]Set [b]bDLC=True[/b] to remove DLC items.
|
||||||
[/list]
|
[/list]
|
||||||
|
|
||||||
|
[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-LootedTraderInventory]https://github.com/GenZmeY/KF2-LootedTraderInventory[/url] [b](GNU GPLv3)[/b]
|
[url=https://github.com/GenZmeY/KF2-LootedTraderInventory]https://github.com/GenZmeY/KF2-LootedTraderInventory[/url] [b](GNU GPLv3)[/b]
|
||||||
|
12
README.md
12
README.md
@ -3,17 +3,17 @@
|
|||||||
[](https://steamcommunity.com/sharedfiles/filedetails/?id=2864857909)
|
[](https://steamcommunity.com/sharedfiles/filedetails/?id=2864857909)
|
||||||
[](https://steamcommunity.com/sharedfiles/filedetails/?id=2864857909)
|
[](https://steamcommunity.com/sharedfiles/filedetails/?id=2864857909)
|
||||||
[](https://steamcommunity.com/sharedfiles/filedetails/?id=2864857909)
|
[](https://steamcommunity.com/sharedfiles/filedetails/?id=2864857909)
|
||||||
[](https://steamcommunity.com/sharedfiles/filedetails/?id=2864857909)
|
[](https://github.com/GenZmeY/KF2-LootedTraderInventory/actions/workflows/mega-linter.yml)
|
||||||
[](https://github.com/GenZmeY/KF2-LootedTraderInventory/tags)
|
[](https://github.com/GenZmeY/KF2-LootedTraderInventory/tags)
|
||||||
[](LICENSE)
|
[](LICENSE)
|
||||||
|
|
||||||
# Description
|
## Description
|
||||||
This is a heavily stripped down version of [CTI](https://github.com/GenZmeY/KF2-CustomTraderInventory) that only allows you to remove the trader's weapons, not add them. This only exists in hopes of being whitelisted.
|
This is a heavily stripped down version of [CTI](https://github.com/GenZmeY/KF2-CustomTraderInventory) that only allows you to remove the trader's weapons, not add them. This only exists in hopes of being whitelisted.
|
||||||
|
|
||||||
# Usage & Setup
|
## Usage & Setup
|
||||||
[See steam workshop page](https://steamcommunity.com/sharedfiles/filedetails/?id=2864857909)
|
[See steam workshop page](https://steamcommunity.com/sharedfiles/filedetails/?id=2864857909)
|
||||||
|
|
||||||
# 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:
|
||||||
@ -27,5 +27,5 @@ This is a heavily stripped down version of [CTI](https://github.com/GenZmeY/KF2-
|
|||||||
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\`
|
||||||
|
|
||||||
# License
|
## License
|
||||||
[GNU GPLv3](LICENSE)
|
[](LICENSE)
|
||||||
|
@ -7,7 +7,7 @@ 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="LTI"
|
PackageBuildOrder="LTI"
|
||||||
|
|
||||||
|
|
||||||
@ -16,7 +16,7 @@ PackageBuildOrder="LTI"
|
|||||||
# Packages you want to brew using @peelz's patched KFEditor.
|
# Packages you want to brew using @peelz's patched KFEditor.
|
||||||
# Useful for cases where regular brew doesn't put *.upk inside the package.
|
# Useful for cases where regular brew doesn't put *.upk inside the package.
|
||||||
# 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
|
||||||
PackagePeelzBrew=""
|
PackagePeelzBrew=""
|
||||||
|
|
||||||
|
|
||||||
@ -24,7 +24,7 @@ PackagePeelzBrew=""
|
|||||||
|
|
||||||
# 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="LTI"
|
PackageUpload="LTI"
|
||||||
|
|
||||||
|
|
||||||
|
2
tools
2
tools
@ -1 +1 @@
|
|||||||
Subproject commit 0e821f3dbbc6b3528f2028b0060d3b6f7f1c4b93
|
Subproject commit fb458ac61f7e6c6426b8dff366dd5e7499e0d95f
|
Loading…
x
Reference in New Issue
Block a user