1608 lines
46 KiB
Ucode
Raw Normal View History

2017-10-19 21:00:49 -05:00
// Base perk.
Class Ext_PerkBase extends Info
NotPlaceable
Abstract
Config(ServerExt)
DependsOn(ExtWebAdmin_UI);
var array<FWebAdminConfigInfo> WebConfigs;
var ExtPerkManager PerkManager;
var Controller PlayerOwner;
var() localized string PerkName;
2017-10-19 21:00:49 -05:00
var() Texture2D PerkIcon;
var() class<KFPerk> BasePerk; // KF perk that this perk is based on.
var() class<KFWeapon> PrimaryMelee,PrimaryWeapon;
var() class<KFWeaponDefinition> PrimaryWeaponDef,SecondaryWeaponDef,KnifeWeaponDef,GrenadeWeaponDef;
var() class<KFProj_Grenade> GrenadeClass,PerkGrenade,SuperGrenade;
var() int HealExpUpNum,WeldExpUpNum; // Efficiency of healing and welding XP up.
// For trader.
var() array<class<KFWeaponDefinition> > AutoBuyLoadOutPath;
// Config init stuff.
var config int ConfigVersion;
var const int CurrentConfigVer;
// Variables.
var config int FirstLevelExp, // How much EXP needed for first level.
LevelUpExpCost, // How much EXP needed for every level up.
LevelUpIncCost, // How much EXP increase needed for each level up.
MinimumLevel,
MaximumLevel,
StarPointsPerLevel,
MinLevelForPrestige, // Minimum level required for perk prestige.
PrestigeSPIncrease, // Starpoint increase per prestige levelup.
MaxPrestige,
MinimalDataLevel; // Maximum prestige level.
2017-10-19 21:00:49 -05:00
var config float PrestigeXPReduce; // Amount of XP cost is reduced for each prestige.
var config array<string> TraitClasses;
var array<float> Modifiers;
var int CurrentLevel, // Current level player is on.
CurrentEXP, // Current amount of EXP user has.
NextLevelEXP, // Experience needed for next level.
CurrentSP, // Current amount of star points.
LastLevelEXP, // Number of XP was needed for last level.
CurrentPrestige; // Current prestige level.
struct FPerkStat
{
var config int MaxValue,CostPerValue;
var config float Progress;
var config name StatType;
var transient int CurrentValue,OldValue;
var transient float DisplayValue;
var transient string UIName;
};
var config array<FPerkStat> PerkStats;
struct FDefPerkStat
{
var int MaxValue,CostPerValue;
var float Progress;
var name StatType;
var bool bHiddenConfig; // Hide this config by default.
};
var() array<FDefPerkStat> DefPerkStats;
var() array< class<Ext_TraitBase> > DefTraitList;
struct FPlayerTrait
{
var class<Ext_TraitBase> TraitType;
var byte CurrentLevel;
var Ext_TraitDataStore Data;
};
var array<FPlayerTrait> PerkTraits;
// Server -> Client replication variables
var byte RepState;
var int RepIndex;
var transient float NextAuthTime;
var int ToxicDartDamage;
var byte EnemyHealthRange;
var() array<float> EnemyDistDraw;
var bool bOwnerNetClient,bClientAuthorized,bPerkNetReady,bHasNightVision,bCanBeGrabbed,bExplosiveWeld,bExplodeOnContact,bNapalmFire,bFireExplode,bToxicDart,bTacticalReload,bHeavyArmor,bHasSWATEnforcer;
2020-09-01 11:07:53 +03:00
var localized string StatSpeed;
var localized string StatDamage;
var localized string StatRecoil;
var localized string StatSpread;
var localized string StatRate;
var localized string StatReload;
var localized string StatHealth;
var localized string StatKnockDown;
var localized string StatWelder;
var localized string StatHeal;
var localized string StatMag;
var localized string StatSpare;
var localized string StatOffDamage;
var localized string StatSelfDamage;
var localized string StatArmor;
var localized string StatPoisonDmg;
var localized string StatSonicDmg;
var localized string StatFireDmg;
var localized string StatAllDmg;
var localized string StatHeadDamage;
var localized string StatHealRecharge;
2021-08-21 10:53:27 -07:00
var localized string StatSwitch;
2020-09-01 11:07:53 +03:00
2020-09-07 14:49:46 +03:00
reliable client simulated function string UIName(FDefPerkStat DefPerkStat)
2020-09-01 11:07:53 +03:00
{
2020-11-28 23:12:58 +03:00
switch (DefPerkStat.StatType)
2020-09-01 11:07:53 +03:00
{
2020-11-28 22:53:57 +03:00
case name("Speed"): return StatSpeed;
case name("Damage"): return StatDamage;
case name("Recoil"): return StatRecoil;
case name("Spread"): return StatSpread;
case name("Rate"): return StatRate;
case name("Reload"): return StatReload;
case name("Health"): return StatHealth;
case name("KnockDown"): return StatKnockDown;
case name("Welder"): return StatWelder;
case name("Heal"): return StatHeal;
case name("Mag"): return StatMag;
case name("Spare"): return StatSpare;
case name("OffDamage"): return StatOffDamage;
2020-09-01 11:07:53 +03:00
case name("SelfDamage"): return StatSelfDamage;
2020-11-28 22:53:57 +03:00
case name("Armor"): return StatArmor;
case name("PoisonDmg"): return StatPoisonDmg;
case name("SonicDmg"): return StatSonicDmg;
case name("FireDmg"): return StatFireDmg;
case name("AllDmg"): return StatAllDmg;
2020-09-01 11:07:53 +03:00
case name("HeadDamage"): return StatHeadDamage;
case name("HealRecharge"): return StatHealRecharge;
2021-08-21 10:53:27 -07:00
case name("Switch"): return StatSwitch;
2020-09-01 11:07:53 +03:00
}
return "";
}
2017-10-19 21:00:49 -05:00
replication
{
// Things the server should send to the client.
2020-11-28 23:04:55 +03:00
if (true)
2017-10-19 21:00:49 -05:00
CurrentLevel,CurrentPrestige,CurrentEXP,NextLevelEXP,CurrentSP,LastLevelEXP,bHasNightVision,MinLevelForPrestige,PrestigeSPIncrease,MaxPrestige,bTacticalReload,EnemyHealthRange;
}
2020-11-28 23:04:55 +03:00
simulated final function bool IsWeaponOnPerk(KFWeapon W)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (class<KFPerk_Survivalist>(BasePerk) != None)
return true;
2020-11-28 23:12:58 +03:00
//if (W.AllowedForAllPerks())
// return true;
2023-05-14 05:49:12 +03:00
2017-10-19 21:00:49 -05:00
return W!=None && W.GetWeaponPerkClass(BasePerk)==BasePerk;
}
2020-11-28 23:04:55 +03:00
simulated static function string GetPerkIconPath(int Level)
2017-10-19 21:00:49 -05:00
{
2020-01-09 05:05:13 -06:00
return "img://"$PathName(default.PerkIcon);
2017-10-19 21:00:49 -05:00
}
simulated function PostBeginPlay()
{
local int i,j;
local class<Ext_TraitBase> T;
2020-11-28 23:12:58 +03:00
if (WorldInfo.NetMode==NM_Client)
2017-10-19 21:00:49 -05:00
{
PerkStats.Length = 0; // Prevent client desync with client settings.
PlayerOwner = GetALocalPlayerController();
SetTimer(0.01,false,'InitPerk');
}
else
{
RemoteRole = ROLE_None; // Make sure these actors get replicated in order to client.
PlayerOwner = Controller(Owner);
2020-11-28 23:12:58 +03:00
if (PlayerOwner==None)
2017-10-19 21:00:49 -05:00
{
`Log(Self@"spawned without owner.");
Destroy();
return;
}
bOwnerNetClient = (PlayerController(Owner)!=None && LocalPlayer(PlayerController(Owner).Player)==None);
2023-05-14 05:49:12 +03:00
2017-10-19 21:00:49 -05:00
// Load trait classes.
j = 0;
2020-11-28 23:12:58 +03:00
for (i=0; i<TraitClasses.Length; ++i)
2017-10-19 21:00:49 -05:00
{
T = class<Ext_TraitBase>(DynamicLoadObject(TraitClasses[i],Class'Class'));
2020-11-28 23:12:58 +03:00
if (T==None || !T.Static.IsEnabled(Self))
2017-10-19 21:00:49 -05:00
continue;
PerkTraits.Length = j+1;
PerkTraits[j].TraitType = T;
++j;
}
2023-05-14 05:49:12 +03:00
2017-10-19 21:00:49 -05:00
// Setup serverside stat info (for XML log files).
2020-11-28 23:12:58 +03:00
for (j=0; j<PerkStats.Length; ++j)
2017-10-19 21:00:49 -05:00
{
i = DefPerkStats.Find('StatType',PerkStats[j].StatType);
2020-11-28 23:12:58 +03:00
if (i>=0)
2020-09-01 11:07:53 +03:00
PerkStats[j].UIName = UIName(DefPerkStats[i]);
2017-10-19 21:00:49 -05:00
else
{
// Fallback to parent perk for trying to find name.
i = Class'Ext_PerkBase'.Default.DefPerkStats.Find('StatType',PerkStats[j].StatType);
2020-11-28 23:12:58 +03:00
if (i>=0)
2020-09-01 11:07:53 +03:00
PerkStats[j].UIName = UIName(Class'Ext_PerkBase'.Default.DefPerkStats[i]);
2017-10-19 21:00:49 -05:00
else PerkStats[j].UIName = string(PerkStats[j].StatType); // Fallback to stat name then...
}
}
}
}
2020-11-29 00:54:57 +03:00
2017-10-19 21:00:49 -05:00
simulated function InitPerk()
{
2020-11-28 23:12:58 +03:00
if (PlayerOwner==None)
2017-10-19 21:00:49 -05:00
PlayerOwner = GetALocalPlayerController();
2020-11-28 23:12:58 +03:00
if (PerkManager==None)
2017-10-19 21:00:49 -05:00
{
foreach DynamicActors(class'ExtPerkManager',PerkManager)
{
PerkManager.RegisterPerk(Self);
break;
}
}
}
2020-11-29 00:54:57 +03:00
2017-10-19 21:00:49 -05:00
simulated function Destroyed()
{
local int i;
2020-11-28 23:12:58 +03:00
if (PerkManager!=None)
2017-10-19 21:00:49 -05:00
PerkManager.UnregisterPerk(Self);
2020-11-28 23:12:58 +03:00
if (WorldInfo.NetMode!=NM_Client)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
for (i=0; i<PerkTraits.Length; ++i)
2017-10-19 21:00:49 -05:00
PerkTraits[i].TraitType.Static.CleanupTrait(ExtPlayerController(Owner),Self,PerkTraits[i].Data);
}
}
// For HUD UI
simulated final function string GetLevelString()
{
return (CurrentPrestige>0 ? (string(CurrentPrestige)$"-"$string(CurrentLevel)) : string(CurrentLevel));
}
// For progress bar on HUD
simulated final function float GetProgressPercent()
{
return FClamp(float(CurrentEXP-LastLevelEXP) / FMax(float(NextLevelEXP-LastLevelEXP),1.f),0.f,1.f);
}
// Whetever if user can use prestige now.
simulated final function bool CanPrestige()
{
return (MinLevelForPrestige>=0 && CurrentPrestige<MaxPrestige) ? CurrentLevel>=MinLevelForPrestige : false;
}
// Whetever to save this perk status or not
final function bool HasAnyProgress()
{
return (CurrentEXP>0 || CurrentPrestige>0);
}
2020-11-28 23:04:55 +03:00
reliable client simulated function ClientReceiveStat(int Index, int MaxValue, int CostPerValue, name Type, int CurValue, float Progress)
2017-10-19 21:00:49 -05:00
{
local int i;
2020-11-28 23:12:58 +03:00
if (WorldInfo.NetMode==NM_Client)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (PerkStats.Length<=Index)
2017-10-19 21:00:49 -05:00
PerkStats.Length = Index+1;
PerkStats[Index].MaxValue = MaxValue;
PerkStats[Index].CostPerValue = CostPerValue;
PerkStats[Index].StatType = Type;
PerkStats[Index].CurrentValue = CurValue;
PerkStats[Index].DisplayValue = 0.f;
PerkStats[Index].Progress = Progress;
}
i = DefPerkStats.Find('StatType',Type);
2020-11-28 23:12:58 +03:00
if (i>=0)
2020-09-01 11:07:53 +03:00
PerkStats[Index].UIName = UIName(DefPerkStats[i]);
2017-10-19 21:00:49 -05:00
else
{
// Fallback to parent perk for trying to find name.
i = Class'Ext_PerkBase'.Default.DefPerkStats.Find('StatType',Type);
2020-11-28 23:12:58 +03:00
if (i>=0)
2020-09-01 11:07:53 +03:00
PerkStats[Index].UIName = UIName(Class'Ext_PerkBase'.Default.DefPerkStats[i]);
2017-10-19 21:00:49 -05:00
else PerkStats[Index].UIName = string(Type); // Fallback to stat name then...
}
}
2020-11-29 00:54:57 +03:00
2020-11-28 23:04:55 +03:00
reliable client simulated function ClientSetStatValue(int Index, int NewValue)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (PerkStats.Length<=Index)
2017-10-19 21:00:49 -05:00
PerkStats.Length = Index+1;
PerkStats[Index].CurrentValue = NewValue;
2020-11-28 23:12:58 +03:00
if (bPerkNetReady)
2017-10-19 21:00:49 -05:00
ApplyEffects();
}
2020-11-29 00:54:57 +03:00
2020-11-28 23:04:55 +03:00
reliable client simulated function ClientReceiveTrait(int Index, class<Ext_TraitBase> TC, byte Lvl)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (PerkTraits.Length<=Index)
2017-10-19 21:00:49 -05:00
PerkTraits.Length = Index+1;
PerkTraits[Index].TraitType = TC;
PerkTraits[Index].CurrentLevel = Lvl;
}
2020-11-29 00:54:57 +03:00
2020-11-28 23:04:55 +03:00
reliable client simulated function ClientReceiveTraitData(int Index, string Data)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (WorldInfo.NetMode==NM_Client)
2017-10-19 21:00:49 -05:00
PerkTraits[Index].TraitType.Static.ClientSetRepData(Data);
}
2020-11-29 00:54:57 +03:00
2020-11-28 23:04:55 +03:00
reliable client simulated function ClientReceiveTraitLvl(int Index, byte NewLevel)
2017-10-19 21:00:49 -05:00
{
PerkTraits[Index].CurrentLevel = NewLevel;
}
2020-11-28 23:04:55 +03:00
final function SetPerkStat(name Type, int Value)
2017-10-19 21:00:49 -05:00
{
local int i;
2023-05-14 05:49:12 +03:00
2017-10-19 21:00:49 -05:00
i = PerkStats.Find('StatType',Type);
2020-11-28 23:12:58 +03:00
if (i>=0)
2017-10-19 21:00:49 -05:00
PerkStats[i].CurrentValue = Value;
}
2020-11-29 00:54:57 +03:00
2020-11-28 23:04:55 +03:00
final function int GetPerkStat(name Type)
2017-10-19 21:00:49 -05:00
{
local int i;
2023-05-14 05:49:12 +03:00
2017-10-19 21:00:49 -05:00
i = PerkStats.Find('StatType',Type);
2020-11-28 23:12:58 +03:00
if (i==-1)
2017-10-19 21:00:49 -05:00
return 0;
return PerkStats[i].CurrentValue;
}
2020-11-28 23:04:55 +03:00
function bool EarnedEXP(int EXP)
2017-10-19 21:00:49 -05:00
{
local int n;
bForceNetUpdate = true;
CurrentEXP+=EXP;
2020-11-28 23:12:58 +03:00
while (CurrentEXP>=NextLevelEXP && CurrentLevel<MaximumLevel && n<20)
2017-10-19 21:00:49 -05:00
{
++n;
LastLevelEXP = NextLevelEXP;
NextLevelEXP = GetNeededExp(++CurrentLevel);
}
2020-11-28 23:12:58 +03:00
if (n>0)
2017-10-19 21:00:49 -05:00
{
CurrentSP+=(n*(StarPointsPerLevel+CurrentPrestige*PrestigeSPIncrease));
2020-11-28 23:12:58 +03:00
if (PerkManager.PRIOwner!=None && PerkManager.CurrentPerk==Self)
2017-10-19 21:00:49 -05:00
UpdatePRILevel();
// TODO - broadcast level up messages.
2020-11-28 23:12:58 +03:00
if (ExtPlayerController(PlayerOwner)!=None)
2017-10-19 21:00:49 -05:00
ExtPlayerController(PlayerOwner).ReceiveLevelUp(Self,CurrentLevel);
}
return true;
}
final function UpdatePRILevel()
{
PerkManager.PRIOwner.SetLevelProgress(CurrentLevel,CurrentPrestige,MinimumLevel,MaximumLevel);
}
// XML output
2020-11-28 23:04:55 +03:00
function OutputXML(ExtStatWriter Data)
2017-10-19 21:00:49 -05:00
{
local int i;
Data.StartIntendent("perk","class",string(Class.Name));
Data.WriteValue("perkname",PerkName);
Data.WriteValue("level",string(CurrentLevel));
Data.WriteValue("prestige",string(CurrentPrestige));
Data.WriteValue("exp",string(CurrentEXP));
Data.WriteValue("points",string(CurrentSP));
Data.WriteValue("exptilnext",string(NextLevelEXP));
Data.WriteValue("exponprev",string(LastLevelEXP));
2023-05-14 05:49:12 +03:00
2020-11-28 23:12:58 +03:00
for (i=0; i<PerkStats.Length; ++i)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (PerkStats[i].CurrentValue>0)
2017-10-19 21:00:49 -05:00
{
Data.StartIntendent("stat","type",string(PerkStats[i].StatType));
Data.WriteValue("name",GetStatUIStr(i));
Data.WriteValue("value",string(PerkStats[i].CurrentValue));
Data.WriteValue("progress",string(PerkStats[i].DisplayValue));
Data.EndIntendent();
}
}
2023-05-14 05:49:12 +03:00
2020-11-28 23:12:58 +03:00
for (i=0; i<PerkTraits.Length; ++i)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (PerkTraits[i].CurrentLevel>0)
2017-10-19 21:00:49 -05:00
{
Data.StartIntendent("trait","class",string(PerkTraits[i].TraitType.Name));
Data.WriteValue("name",PerkTraits[i].TraitType.Default.TraitName);
Data.WriteValue("level",string(PerkTraits[i].CurrentLevel));
Data.EndIntendent();
}
}
Data.EndIntendent();
}
// Data saving.
2020-11-28 23:04:55 +03:00
function SaveData(ExtSaveDataBase Data)
2017-10-19 21:00:49 -05:00
{
local int i,j;
2023-05-14 05:49:12 +03:00
2017-10-19 21:00:49 -05:00
// Write current EXP.
Data.SaveInt(CurrentEXP,3);
2023-05-14 05:49:12 +03:00
2017-10-19 21:00:49 -05:00
// Write current prestige
Data.SaveInt(CurrentPrestige,3);
// Count number of given stats
j = 0;
2020-11-28 23:12:58 +03:00
for (i=0; i<PerkStats.Length; ++i)
if (PerkStats[i].CurrentValue>0)
2017-10-19 21:00:49 -05:00
++j;
// Then perk stats.
Data.SaveInt(j);
2020-11-28 23:12:58 +03:00
for (i=0; i<PerkStats.Length; ++i)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (PerkStats[i].CurrentValue>0)
2017-10-19 21:00:49 -05:00
{
Data.SaveStr(string(PerkStats[i].StatType));
Data.SaveInt(PerkStats[i].CurrentValue,1);
}
}
2023-05-14 05:49:12 +03:00
2017-10-19 21:00:49 -05:00
// Count bought traits.
j = 0;
2020-11-28 23:12:58 +03:00
for (i=0; i<PerkTraits.Length; ++i)
if (PerkTraits[i].CurrentLevel>0)
2017-10-19 21:00:49 -05:00
++j;
// Then traits.
Data.SaveInt(j);
2020-11-28 23:12:58 +03:00
for (i=0; i<PerkTraits.Length; ++i)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (PerkTraits[i].CurrentLevel>0)
2017-10-19 21:00:49 -05:00
{
Data.SaveStr(string(PerkTraits[i].TraitType));
Data.SaveInt(PerkTraits[i].CurrentLevel);
}
}
}
// Data loading.
2020-11-28 23:04:55 +03:00
function LoadData(ExtSaveDataBase Data)
2017-10-19 21:00:49 -05:00
{
local int i,j,l,n;
local string S;
CurrentEXP = Data.ReadInt(3);
2020-11-28 23:12:58 +03:00
// if (MinimalDataLevel > 0)
// {
// i = GetNeededExp(MinimalDataLevel-1)
2020-11-28 23:12:58 +03:00
// if (i > CurrentEXP)
// CurrentEXP = i
// }
2023-05-14 05:49:12 +03:00
2020-11-28 23:12:58 +03:00
if (Data.GetArVer()>=1)
2017-10-19 21:00:49 -05:00
CurrentPrestige = Data.ReadInt(3);
l = Data.ReadInt(); // Perk stats length.
2020-11-28 23:12:58 +03:00
for (i=0; i<l; ++i)
2017-10-19 21:00:49 -05:00
{
S = Data.ReadStr();
n = Data.ReadInt(1);
2020-11-28 23:12:58 +03:00
for (j=0; j<PerkStats.Length; ++j)
if (S~=string(PerkStats[j].StatType))
2017-10-19 21:00:49 -05:00
{
PerkStats[j].CurrentValue = n;
break;
}
}
2023-05-14 05:49:12 +03:00
2017-10-19 21:00:49 -05:00
l = Data.ReadInt(); // Traits stats length.
2020-11-28 23:12:58 +03:00
for (i=0; i<l; ++i)
2017-10-19 21:00:49 -05:00
{
S = Data.ReadStr();
n = Data.ReadInt();
2020-11-28 23:12:58 +03:00
for (j=0; j<PerkTraits.Length; ++j)
if (S~=string(PerkTraits[j].TraitType))
2017-10-19 21:00:49 -05:00
{
PerkTraits[j].CurrentLevel = n;
break;
}
}
}
2020-11-28 23:04:55 +03:00
final function int CalcLevelForExp(int InExp)
2017-10-19 21:00:49 -05:00
{
local int i,a,b;
// Fast method to calc level for a player.
b = MaximumLevel+1;
a = Min(MinimumLevel,b);
2020-11-28 23:12:58 +03:00
while (true)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (a==b || (a+1)==b)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (a<MaximumLevel && InExp>=GetNeededExp(a))
2017-10-19 21:00:49 -05:00
++a;
break;
}
i = a+((b-a)>>1);
2020-11-28 23:12:58 +03:00
if (InExp<GetNeededExp(i)) // Lower!
2017-10-19 21:00:49 -05:00
b = i;
else a = i; // Higher!
}
return Clamp(a,MinimumLevel,MaximumLevel);
}
// Initialize perk after stats have been loaded.
function SetInitialLevel()
{
local int i,a,b;
local byte MT,j;
2020-11-28 23:12:58 +03:00
if (MinimalDataLevel > 0)
{
i = GetNeededExp(MinimalDataLevel-1);
2020-11-28 23:12:58 +03:00
if (i > CurrentEXP)
CurrentEXP = i;
}
2017-10-19 21:00:49 -05:00
// Set to initial level player is on after configures has loaded.
CurrentLevel = CalcLevelForExp(CurrentEXP);
CurrentSP = CurrentLevel*(StarPointsPerLevel+CurrentPrestige*PrestigeSPIncrease);
NextLevelEXP = GetNeededExp(CurrentLevel);
LastLevelEXP = (CurrentLevel>MinimumLevel ? GetNeededExp(CurrentLevel-1) : 0);
2023-05-14 05:49:12 +03:00
2017-10-19 21:00:49 -05:00
// Now verify the points player used on individual stats.
2020-11-28 23:12:58 +03:00
for (i=0; i<PerkStats.Length; ++i)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (PerkStats[i].CurrentValue>0)
2017-10-19 21:00:49 -05:00
{
PerkStats[i].CurrentValue = Min(PerkStats[i].CurrentValue,PerkStats[i].MaxValue);
a = PerkStats[i].CurrentValue*PerkStats[i].CostPerValue;
2020-11-28 23:12:58 +03:00
if (CurrentSP>a)
2017-10-19 21:00:49 -05:00
CurrentSP-=a;
2020-11-28 23:12:58 +03:00
else if (CurrentSP<=0) // No points at all for this.
2017-10-19 21:00:49 -05:00
PerkStats[i].CurrentValue = 0;
else // Nope, reduce the stat!
{
a = CurrentSP/PerkStats[i].CostPerValue;
PerkStats[i].CurrentValue = a;
CurrentSP-=(a*PerkStats[i].CostPerValue);
}
}
}
// Then verify trait levels and costs.
MT = 0;
2020-11-28 23:12:58 +03:00
for (i=0; i<PerkTraits.Length; ++i)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (PerkTraits[i].CurrentLevel>0)
2017-10-19 21:00:49 -05:00
{
PerkTraits[i].CurrentLevel = Min(PerkTraits[i].CurrentLevel,PerkTraits[i].TraitType.Default.NumLevels);
2023-05-14 05:49:12 +03:00
2020-11-28 23:12:58 +03:00
if (PerkTraits[i].TraitType.Default.LoadPriority>0)
2017-10-19 21:00:49 -05:00
MT = Max(MT,PerkTraits[i].TraitType.Default.LoadPriority);
else
{
2020-11-28 23:12:58 +03:00
if (!PerkTraits[i].TraitType.Static.MeetsRequirements(PerkTraits[i].CurrentLevel-1,Self))
2017-10-19 21:00:49 -05:00
a = 0;
else
{
2020-11-28 23:12:58 +03:00
for (a=0; a<PerkTraits[i].CurrentLevel; ++a)
2017-10-19 21:00:49 -05:00
{
b = PerkTraits[i].TraitType.Static.GetTraitCost(a);
2020-11-28 23:12:58 +03:00
if (b>CurrentSP)
2017-10-19 21:00:49 -05:00
break;
CurrentSP-=b;
}
}
PerkTraits[i].CurrentLevel = a;
2020-11-28 23:12:58 +03:00
if (PerkTraits[i].CurrentLevel>0 && PerkTraits[i].Data==None)
PerkTraits[i].Data = PerkTraits[i].TraitType.Static.Initializefor (Self,ExtPlayerController(Owner));
2017-10-19 21:00:49 -05:00
}
}
2020-11-28 23:12:58 +03:00
if (PerkTraits[i].CurrentLevel==0 && PerkTraits[i].Data!=None)
2017-10-19 21:00:49 -05:00
PerkTraits[i].TraitType.Static.CleanupTrait(ExtPlayerController(Owner),Self,PerkTraits[i].Data);
}
2023-05-14 05:49:12 +03:00
2017-10-19 21:00:49 -05:00
// Delayed loads.
2020-11-28 23:12:58 +03:00
for (j=1; j<=MT; ++j)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
for (i=0; i<PerkTraits.Length; ++i)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (PerkTraits[i].CurrentLevel>0 && PerkTraits[i].TraitType.Default.LoadPriority==j)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (!PerkTraits[i].TraitType.Static.MeetsRequirements(PerkTraits[i].CurrentLevel-1,Self))
2017-10-19 21:00:49 -05:00
a = 0;
else
{
2020-11-28 23:12:58 +03:00
for (a=0; a<PerkTraits[i].CurrentLevel; ++a)
2017-10-19 21:00:49 -05:00
{
b = PerkTraits[i].TraitType.Static.GetTraitCost(a);
2020-11-28 23:12:58 +03:00
if (b>CurrentSP)
2017-10-19 21:00:49 -05:00
break;
CurrentSP-=b;
}
}
PerkTraits[i].CurrentLevel = a;
2020-11-28 23:12:58 +03:00
if (PerkTraits[i].CurrentLevel>0 && PerkTraits[i].Data==None)
PerkTraits[i].Data = PerkTraits[i].TraitType.Static.Initializefor (Self,ExtPlayerController(Owner));
2017-10-19 21:00:49 -05:00
}
2020-11-28 23:12:58 +03:00
if (PerkTraits[i].CurrentLevel==0 && PerkTraits[i].Data!=None)
2017-10-19 21:00:49 -05:00
PerkTraits[i].TraitType.Static.CleanupTrait(ExtPlayerController(Owner),Self,PerkTraits[i].Data);
}
}
ApplyEffects();
2020-11-28 23:12:58 +03:00
if (PerkManager.CurrentPerk==Self)
2017-10-19 21:00:49 -05:00
ActivateTraits();
}
// Check the needed amount of EXP for a perk.
2020-11-28 23:04:55 +03:00
function int GetNeededExp(int LevelNum)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (LevelNum<MinimumLevel || LevelNum>=MaximumLevel)
2017-10-19 21:00:49 -05:00
return 0;
LevelNum-=MinimumLevel;
LevelNum = (FirstLevelExp+(LevelNum*LevelUpExpCost)+(LevelNum*LevelNum*LevelUpIncCost));
2020-11-28 23:12:58 +03:00
if (CurrentPrestige>0 && PrestigeXPReduce>0)
2017-10-19 21:00:49 -05:00
LevelNum *= (1.f / (1.f + PrestigeXPReduce*CurrentPrestige));
return LevelNum;
}
// Configure initialization.
static function CheckConfig()
{
local int i;
local class<Ext_TraitBase> T;
2020-11-28 23:12:58 +03:00
if (Default.ConfigVersion!=Default.CurrentConfigVer)
2017-10-19 21:00:49 -05:00
{
UpdateConfigs(Default.ConfigVersion);
Default.ConfigVersion = Default.CurrentConfigVer;
StaticSaveConfig();
}
2020-11-28 23:12:58 +03:00
for (i=0; i<Default.TraitClasses.Length; ++i)
2017-10-19 21:00:49 -05:00
{
T = class<Ext_TraitBase>(DynamicLoadObject(Default.TraitClasses[i],Class'Class'));
2020-11-28 23:12:58 +03:00
if (T!=None)
2017-10-19 21:00:49 -05:00
T.Static.CheckConfig();
}
}
2020-11-29 00:54:57 +03:00
2020-11-28 23:04:55 +03:00
static function UpdateConfigs(int OldVer)
2017-10-19 21:00:49 -05:00
{
local int i,j;
2020-11-28 23:12:58 +03:00
if (OldVer==0)
2017-10-19 21:00:49 -05:00
{
Default.FirstLevelExp = 400;
Default.LevelUpExpCost = 500;
Default.LevelUpIncCost = 65;
Default.MinimumLevel = 0;
Default.MaximumLevel = 150;
Default.StarPointsPerLevel = 15;
2023-05-14 05:49:12 +03:00
2017-10-19 21:00:49 -05:00
// Prestige.
Default.MinLevelForPrestige = 140;
Default.PrestigeSPIncrease = 1;
Default.MaxPrestige = 20;
Default.MinimalDataLevel = 0;
2017-10-19 21:00:49 -05:00
Default.PrestigeXPReduce = 0.05;
2023-05-14 05:49:12 +03:00
2017-10-19 21:00:49 -05:00
Default.PerkStats.Length = 0;
AddStatsCfg(0);
Default.TraitClasses.Length = Default.DefTraitList.Length;
2020-11-28 23:12:58 +03:00
for (i=0; i<Default.DefTraitList.Length; ++i)
2017-10-19 21:00:49 -05:00
Default.TraitClasses[i] = PathName(Default.DefTraitList[i]);
}
else
{
// Add progress.
2020-11-28 23:12:58 +03:00
if (OldVer==1)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
for (i=0; i<Default.PerkStats.Length; ++i)
2017-10-19 21:00:49 -05:00
{
j = Default.DefPerkStats.Find('StatType',Default.PerkStats[i].StatType);
2020-11-28 23:12:58 +03:00
if (j>=0)
2017-10-19 21:00:49 -05:00
Default.PerkStats[i].Progress = Default.DefPerkStats[j].Progress;
}
// Add off-perk damage stat.
AddStatsCfg(12);
}
2020-11-28 23:12:58 +03:00
else if (OldVer<=3)
2017-10-19 21:00:49 -05:00
AddStatsCfg(13); // Add self damage.
2020-11-28 23:12:58 +03:00
else if (OldVer<=4)
2017-10-19 21:00:49 -05:00
AddStatsCfg(15); // Add poison damage.
2020-11-28 23:12:58 +03:00
else if (OldVer<=7)
2017-10-19 21:00:49 -05:00
AddStatsCfg(16); // Add sonic/fire damage.
2020-11-28 23:12:58 +03:00
else if (OldVer<=12)
2017-10-19 21:00:49 -05:00
AddStatsCfg(18); // Add all damage.
2020-11-28 23:12:58 +03:00
else if (OldVer<=13)
AddStatsCfg(19); // Add HeadDamage and HealRecharge
else if (OldVer<=14)
AddStatsCfg(21); // Add WeaponSwitch
2020-11-28 23:12:58 +03:00
if (OldVer<=5)
2017-10-19 21:00:49 -05:00
{
// Add prestige
Default.MinLevelForPrestige = 140;
Default.PrestigeSPIncrease = 1;
Default.MaxPrestige = 20;
Default.PrestigeXPReduce = 0.05;
}
Default.TraitClasses.Length = Default.DefTraitList.Length;
2020-11-28 23:12:58 +03:00
for (i=0; i<Default.DefTraitList.Length; ++i)
2017-10-19 21:00:49 -05:00
Default.TraitClasses[i] = PathName(Default.DefTraitList[i]);
}
}
2020-11-29 00:54:57 +03:00
2020-11-28 23:04:55 +03:00
static final function AddStatsCfg(int StartRange)
2017-10-19 21:00:49 -05:00
{
local int i,j;
j = Default.PerkStats.Length;
2020-11-28 23:12:58 +03:00
for (i=StartRange; i<Default.DefPerkStats.Length; ++i)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (Default.DefPerkStats[i].bHiddenConfig || Default.PerkStats.Find('StatType',Default.DefPerkStats[i].StatType)>=0) // Don't add if already found for some reason.
2017-10-19 21:00:49 -05:00
continue;
Default.PerkStats.Length = j+1;
Default.PerkStats[j].MaxValue = Default.DefPerkStats[i].MaxValue;
Default.PerkStats[j].CostPerValue = Default.DefPerkStats[i].CostPerValue;
Default.PerkStats[j].StatType = Default.DefPerkStats[i].StatType;
Default.PerkStats[j].Progress = Default.DefPerkStats[i].Progress;
++j;
}
}
// WebAdmin UI stuff.
2020-11-28 23:04:55 +03:00
static function InitWebAdmin(ExtWebAdmin_UI UI)
2017-10-19 21:00:49 -05:00
{
local class<Ext_TraitBase> T;
local int i;
UI.AddSettingsPage("Perk "$Default.PerkName,Default.Class,Default.WebConfigs,GetValue,ApplyValue);
2023-05-14 05:49:12 +03:00
2020-11-28 23:12:58 +03:00
for (i=0; i<Default.TraitClasses.Length; ++i)
2017-10-19 21:00:49 -05:00
{
T = class<Ext_TraitBase>(DynamicLoadObject(Default.TraitClasses[i],Class'Class'));
2020-11-28 23:12:58 +03:00
if (T==None || UI.HasConfigfor (T))
2017-10-19 21:00:49 -05:00
continue;
T.Static.InitWebAdmin(UI);
}
}
2020-11-29 00:54:57 +03:00
2020-11-28 23:04:55 +03:00
static function string GetValue(name PropName, int ElementIndex)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
switch (PropName)
2017-10-19 21:00:49 -05:00
{
case 'FirstLevelExp':
return string(Default.FirstLevelExp);
case 'LevelUpExpCost':
return string(Default.LevelUpExpCost);
case 'LevelUpIncCost':
return string(Default.LevelUpIncCost);
case 'MinimumLevel':
return string(Default.MinimumLevel);
case 'MaximumLevel':
return string(Default.MaximumLevel);
case 'StarPointsPerLevel':
return string(Default.StarPointsPerLevel);
case 'TraitClasses':
return ElementIndex==-1 ? string(Default.TraitClasses.Length) : Default.TraitClasses[ElementIndex];
case 'PerkStats':
return ElementIndex==-1 ? string(Default.PerkStats.Length) : Default.PerkStats[ElementIndex].StatType$","$Default.PerkStats[ElementIndex].MaxValue$","$Default.PerkStats[ElementIndex].CostPerValue$","$Default.PerkStats[ElementIndex].Progress;
case 'MinLevelForPrestige':
return string(Default.MinLevelForPrestige);
case 'PrestigeSPIncrease':
return string(Default.PrestigeSPIncrease);
case 'MinimalDataLevel':
return string(Default.MinimalDataLevel);
2017-10-19 21:00:49 -05:00
case 'MaxPrestige':
return string(Default.MaxPrestige);
case 'PrestigeXPReduce':
return string(Default.PrestigeXPReduce);
}
}
2020-11-29 00:54:57 +03:00
2020-11-28 23:04:55 +03:00
static function ApplyValue(name PropName, int ElementIndex, string Value)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
switch (PropName)
2017-10-19 21:00:49 -05:00
{
case 'FirstLevelExp':
Default.FirstLevelExp = int(Value); break;
case 'LevelUpExpCost':
Default.LevelUpExpCost = int(Value); break;
case 'MinimalDataLevel':
Default.MinimalDataLevel = int(Value); break;
2017-10-19 21:00:49 -05:00
case 'LevelUpIncCost':
Default.LevelUpIncCost = int(Value); break;
case 'MinimumLevel':
Default.MinimumLevel = int(Value); break;
case 'MaximumLevel':
Default.MaximumLevel = int(Value); break;
case 'StarPointsPerLevel':
Default.StarPointsPerLevel = int(Value); break;
case 'TraitClasses':
2020-11-28 23:12:58 +03:00
if (Value=="#DELETE")
2017-10-19 21:00:49 -05:00
Default.TraitClasses.Remove(ElementIndex,1);
else
{
2020-11-28 23:12:58 +03:00
if (ElementIndex>=Default.TraitClasses.Length)
2017-10-19 21:00:49 -05:00
Default.TraitClasses.Length = ElementIndex+1;
Default.TraitClasses[ElementIndex] = Value;
}
break;
case 'PerkStats':
2020-11-28 23:12:58 +03:00
if (Value=="#DELETE")
2017-10-19 21:00:49 -05:00
Default.PerkStats.Remove(ElementIndex,1);
else
{
2020-11-28 23:12:58 +03:00
if (ElementIndex>=Default.PerkStats.Length)
2017-10-19 21:00:49 -05:00
Default.PerkStats.Length = ElementIndex+1;
Default.PerkStats[ElementIndex] = ParsePerkStatStr(Value);
}
break;
case 'MinLevelForPrestige':
Default.MinLevelForPrestige = int(Value); break;
case 'PrestigeSPIncrease':
Default.PrestigeSPIncrease = int(Value); break;
case 'MaxPrestige':
Default.MaxPrestige = int(Value); break;
case 'PrestigeXPReduce':
Default.PrestigeXPReduce = float(Value); break;
default:
return;
}
StaticSaveConfig();
}
2020-11-29 00:54:57 +03:00
2020-11-28 23:04:55 +03:00
static final function FPerkStat ParsePerkStatStr(string S)
2017-10-19 21:00:49 -05:00
{
local FPerkStat Res;
local int i;
i = InStr(S,",");
2020-11-28 23:12:58 +03:00
if (i==-1)
2017-10-19 21:00:49 -05:00
return Res;
Res.StatType = name(Left(S,i));
S = Mid(S,i+1);
i = InStr(S,",");
2020-11-28 23:12:58 +03:00
if (i==-1)
2017-10-19 21:00:49 -05:00
return Res;
Res.MaxValue = int(Left(S,i));
S = Mid(S,i+1);
i = InStr(S,",");
2020-11-28 23:12:58 +03:00
if (i==-1)
2017-10-19 21:00:49 -05:00
return Res;
Res.CostPerValue = int(Left(S,i));
Res.Progress = float(Mid(S,i+1));
return Res;
}
// Amount and iStat values are verified already by ServerExtMut.
2020-11-28 23:04:55 +03:00
function bool IncrementStat(int iStat, int Amount)
2017-10-19 21:00:49 -05:00
{
PerkStats[iStat].CurrentValue+=Amount;
2020-11-28 23:12:58 +03:00
if (bOwnerNetClient)
2017-10-19 21:00:49 -05:00
ClientSetStatValue(iStat,PerkStats[iStat].CurrentValue);
PerkManager.bStatsDirty = true;
ApplyEffects();
bForceNetUpdate = true;
return true;
}
simulated function ApplyEffects()
{
local int i;
2023-05-14 05:49:12 +03:00
2020-11-28 23:12:58 +03:00
for (i=0; i<PerkStats.Length; ++i)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (PerkStats[i].CurrentValue!=PerkStats[i].OldValue)
2017-10-19 21:00:49 -05:00
{
PerkStats[i].OldValue = PerkStats[i].CurrentValue;
PerkStats[i].DisplayValue = ApplyEffect(PerkStats[i].StatType,PerkStats[i].CurrentValue,PerkStats[i].Progress/100.f);
}
}
}
// Notify that player just spawned.
2020-11-28 23:04:55 +03:00
function ApplyEffectsTo(KFPawn_Human P)
2017-10-19 21:00:49 -05:00
{
local int i;
local bool bSec;
2023-05-14 05:49:12 +03:00
2020-11-28 23:12:58 +03:00
for (i=0; i<PerkTraits.Length; ++i)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (PerkTraits[i].CurrentLevel>0)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (PerkTraits[i].TraitType.Default.bPostApplyEffect)
2017-10-19 21:00:49 -05:00
bSec = true;
else PerkTraits[i].TraitType.Static.ApplyEffectOn(P,Self,PerkTraits[i].CurrentLevel,PerkTraits[i].Data);
}
}
2020-11-28 23:12:58 +03:00
if (bSec)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
for (i=0; i<PerkTraits.Length; ++i)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (PerkTraits[i].CurrentLevel>0 && PerkTraits[i].TraitType.Default.bPostApplyEffect)
2017-10-19 21:00:49 -05:00
PerkTraits[i].TraitType.Static.ApplyEffectOn(P,Self,PerkTraits[i].CurrentLevel,PerkTraits[i].Data);
}
}
}
// Player joined/perk changed.
function ActivateTraits()
{
local int i;
local KFPawn_Human KFP;
local bool bSec;
2023-05-14 05:49:12 +03:00
2017-10-19 21:00:49 -05:00
KFP = KFPawn_Human(PlayerOwner.Pawn);
2020-11-28 23:12:58 +03:00
if (KFP!=None && !KFP.IsAliveAndWell())
2017-10-19 21:00:49 -05:00
KFP = None;
2020-11-28 23:12:58 +03:00
for (i=0; i<PerkTraits.Length; ++i)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (PerkTraits[i].CurrentLevel>0)
2017-10-19 21:00:49 -05:00
{
PerkTraits[i].TraitType.Static.TraitActivate(Self,PerkTraits[i].CurrentLevel,PerkTraits[i].Data);
2020-11-28 23:12:58 +03:00
if (KFP!=None)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (PerkTraits[i].TraitType.Default.bPostApplyEffect)
2017-10-19 21:00:49 -05:00
bSec = true;
else PerkTraits[i].TraitType.Static.ApplyEffectOn(KFP,Self,PerkTraits[i].CurrentLevel,PerkTraits[i].Data);
}
}
}
2020-11-28 23:12:58 +03:00
if (bSec)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
for (i=0; i<PerkTraits.Length; ++i)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (PerkTraits[i].CurrentLevel>0 && PerkTraits[i].TraitType.Default.bPostApplyEffect)
2017-10-19 21:00:49 -05:00
PerkTraits[i].TraitType.Static.ApplyEffectOn(KFP,Self,PerkTraits[i].CurrentLevel,PerkTraits[i].Data);
}
}
}
// Player disconnected/perk changed.
function DeactivateTraits()
{
local int i;
2023-05-14 05:49:12 +03:00
2020-11-28 23:12:58 +03:00
for (i=0; i<PerkTraits.Length; ++i)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (PerkTraits[i].CurrentLevel>0)
2017-10-19 21:00:49 -05:00
PerkTraits[i].TraitType.Static.TraitDeActivate(Self,PerkTraits[i].CurrentLevel,PerkTraits[i].Data);
}
}
simulated unreliable client function ClientAuth()
{
2020-11-28 23:12:58 +03:00
if (Owner==None)
2017-10-19 21:00:49 -05:00
SetOwner(PlayerOwner);
ServerAck();
}
2020-11-29 00:54:57 +03:00
2017-10-19 21:00:49 -05:00
unreliable server function ServerAck()
{
2020-11-28 23:12:58 +03:00
if (!bClientAuthorized)
2017-10-19 21:00:49 -05:00
{
bClientAuthorized = true;
RepState = 0;
RepIndex = 0;
SetTimer(0.01+FRand()*0.025,true,'ReplicateTimer');
}
}
2020-11-29 00:54:57 +03:00
2017-10-19 21:00:49 -05:00
function ReplicateTimer()
{
2020-11-28 23:12:58 +03:00
switch (RepState)
2017-10-19 21:00:49 -05:00
{
case 0: // Send all perk stats
2020-11-28 23:12:58 +03:00
if (RepIndex>=PerkStats.Length)
2017-10-19 21:00:49 -05:00
{
++RepState;
RepIndex = 0;
}
else
{
ClientReceiveStat(RepIndex,PerkStats[RepIndex].MaxValue,PerkStats[RepIndex].CostPerValue,PerkStats[RepIndex].StatType,PerkStats[RepIndex].CurrentValue,PerkStats[RepIndex].Progress);
++RepIndex;
}
break;
case 1: // Send all traits
2020-11-28 23:12:58 +03:00
if (RepIndex>=PerkTraits.Length)
2017-10-19 21:00:49 -05:00
++RepState;
else
{
ClientReceiveTrait(RepIndex,PerkTraits[RepIndex].TraitType,PerkTraits[RepIndex].CurrentLevel);
ClientReceiveTraitData(RepIndex,PerkTraits[RepIndex].TraitType.Static.GetRepData());
++RepIndex;
}
break;
default:
ClearTimer('ReplicateTimer');
bPerkNetReady = true;
ClientIsReady(); // Notify client were ready.
}
}
2020-11-29 00:54:57 +03:00
2017-10-19 21:00:49 -05:00
simulated reliable client function ClientIsReady()
{
bPerkNetReady = true;
ApplyEffects();
}
2020-11-29 00:54:57 +03:00
2020-11-28 23:04:55 +03:00
simulated function string GetStatUIStr(int iStat)
2017-10-19 21:00:49 -05:00
{
local string S;
local bool bLoop;
S = string(Abs(PerkStats[iStat].DisplayValue*100.f));
bLoop = true;
// Chop off float digits that aren't needed.
2020-11-28 23:12:58 +03:00
while (bLoop)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
switch (Right(S,1))
2017-10-19 21:00:49 -05:00
{
case "0":
S = Left(S,Len(S)-1);
break;
case ".":
S = Left(S,Len(S)-1);
bLoop = false;
break;
default:
bLoop = false;
}
}
return Repl(PerkStats[iStat].UIName,"&",S);
}
2020-11-28 23:04:55 +03:00
final function UnloadStats(optional byte Mode)
2017-10-19 21:00:49 -05:00
{
local int i,j;
local KFPawn_Human KFP;
PerkManager.bStatsDirty = true;
2020-11-28 23:12:58 +03:00
if (Mode<=1)
2017-10-19 21:00:49 -05:00
{
// Reset stats.
2020-11-28 23:12:58 +03:00
for (i=0; i<PerkStats.Length; ++i)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (PerkStats[i].CurrentValue>0)
2017-10-19 21:00:49 -05:00
{
CurrentSP+=(PerkStats[i].CurrentValue*PerkStats[i].CostPerValue);
PerkStats[i].CurrentValue = 0;
2020-11-28 23:12:58 +03:00
if (bOwnerNetClient)
2017-10-19 21:00:49 -05:00
ClientSetStatValue(i,0);
}
}
ApplyEffects();
}
2020-11-28 23:12:58 +03:00
if (Mode==0 || Mode==2)
2017-10-19 21:00:49 -05:00
{
KFP = KFPawn_Human(PlayerOwner.Pawn);
2020-11-28 23:12:58 +03:00
if (KFP!=None && !KFP.IsAliveAndWell())
2017-10-19 21:00:49 -05:00
KFP = None;
// Reset traits.
2020-11-28 23:12:58 +03:00
for (i=0; i<PerkTraits.Length; ++i)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (PerkTraits[i].CurrentLevel>0)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
for (j=0; j<PerkTraits[i].CurrentLevel; ++j)
2017-10-19 21:00:49 -05:00
CurrentSP+=PerkTraits[i].TraitType.Static.GetTraitCost(j);
2020-11-28 23:12:58 +03:00
if (PerkManager.CurrentPerk==Self)
2017-10-19 21:00:49 -05:00
{
PerkTraits[i].TraitType.Static.TraitDeActivate(Self,PerkTraits[i].CurrentLevel,PerkTraits[i].Data);
2020-11-28 23:12:58 +03:00
if (KFP!=None)
2017-10-19 21:00:49 -05:00
PerkTraits[i].TraitType.Static.CancelEffectOn(KFP,Self,PerkTraits[i].CurrentLevel,PerkTraits[i].Data);
}
PerkTraits[i].TraitType.Static.CleanupTrait(ExtPlayerController(Owner),Self,PerkTraits[i].Data);
PerkTraits[i].CurrentLevel = 0;
2020-11-28 23:12:58 +03:00
if (bOwnerNetClient)
2017-10-19 21:00:49 -05:00
ClientReceiveTraitLvl(i,0);
}
}
}
}
2020-11-29 00:54:57 +03:00
2020-11-28 23:04:55 +03:00
function FullReset(optional bool bNotPrestige)
2017-10-19 21:00:49 -05:00
{
UnloadStats();
// Set minimum values.
CurrentEXP = 0;
2020-11-28 23:12:58 +03:00
if (!bNotPrestige)
2017-10-19 21:00:49 -05:00
CurrentPrestige = 0;
CurrentLevel = MinimumLevel;
CurrentSP = CurrentLevel*(StarPointsPerLevel+CurrentPrestige*PrestigeSPIncrease);
NextLevelEXP = GetNeededExp(CurrentLevel);
LastLevelEXP = 0;
2023-05-14 05:49:12 +03:00
2020-11-28 23:12:58 +03:00
if (PerkManager.CurrentPerk==Self && PerkManager.PRIOwner!=None)
2017-10-19 21:00:49 -05:00
{
PerkManager.PRIOwner.SetLevelProgress(CurrentLevel,CurrentPrestige,MinimumLevel,MaximumLevel);
PerkManager.PRIOwner.bForceNetUpdate = true;
}
bForceNetUpdate = true;
}
2020-11-28 23:04:55 +03:00
function bool PreventDeath(KFPawn_Human Player, Controller Killer, Class<DamageType> DamType)
2017-10-19 21:00:49 -05:00
{
local int i;
2023-05-14 05:49:12 +03:00
2017-10-19 21:00:49 -05:00
// Doing 2 passes of this so that things don't go out of order (spawn retaliation effect when you get redeemed etc)
2020-11-28 23:12:58 +03:00
for (i=0; i<PerkTraits.Length; ++i)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (PerkTraits[i].CurrentLevel>0 && PerkTraits[i].TraitType.Default.bHighPriorityDeath && PerkTraits[i].TraitType.Static.PreventDeath(Player,Killer,DamType,Self,PerkTraits[i].CurrentLevel,PerkTraits[i].Data))
2017-10-19 21:00:49 -05:00
return true;
}
2020-11-28 23:12:58 +03:00
for (i=0; i<PerkTraits.Length; ++i)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (PerkTraits[i].CurrentLevel>0 && !PerkTraits[i].TraitType.Default.bHighPriorityDeath && PerkTraits[i].TraitType.Static.PreventDeath(Player,Killer,DamType,Self,PerkTraits[i].CurrentLevel,PerkTraits[i].Data))
2017-10-19 21:00:49 -05:00
return true;
}
return false;
}
simulated function PlayerDied()
{
local int i;
2020-11-28 23:12:58 +03:00
if (WorldInfo.NetMode!=NM_Client)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
for (i=0; i<PerkTraits.Length; ++i)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (PerkTraits[i].CurrentLevel>0)
2017-10-19 21:00:49 -05:00
PerkTraits[i].TraitType.Static.PlayerDied(Self,PerkTraits[i].CurrentLevel,PerkTraits[i].Data);
}
}
}
// Stat modifier functions.
2020-11-28 23:04:55 +03:00
simulated function float ApplyEffect(name Type, float Value, float Progress)
2017-10-19 21:00:49 -05:00
{
local bool bActivePerk;
2023-05-14 05:49:12 +03:00
2017-10-19 21:00:49 -05:00
bActivePerk = (PerkManager!=None && PerkManager.CurrentPerk==Self);
2020-11-28 23:12:58 +03:00
switch (Type)
2017-10-19 21:00:49 -05:00
{
case 'Speed':
Modifiers[0] = 1.f + (Value*Progress);
break;
case 'Damage':
Modifiers[1] = 1.f + (Value*Progress);
break;
case 'Recoil':
2023-07-17 19:11:20 -07:00
Modifiers[2] = 1.f - (Value*Progress);
2017-10-19 21:00:49 -05:00
break;
case 'Spread':
2023-07-17 19:11:20 -07:00
Modifiers[3] = 1.f - (Value*Progress);
2017-10-19 21:00:49 -05:00
break;
case 'Rate':
Modifiers[4] = 1.f / (1.f+Value*Progress);
break;
case 'Reload':
Modifiers[5] = 1.f / (1.f+Value*Progress);
break;
case 'Health':
Modifiers[6] = 1.f + (Value*Progress);
2020-11-28 23:12:58 +03:00
if (bActivePerk && PlayerOwner.Pawn!=None)
2017-10-19 21:00:49 -05:00
{
PlayerOwner.Pawn.HealthMax = PlayerOwner.Pawn.Default.Health;
ModifyHealth(PlayerOwner.Pawn.HealthMax);
}
break;
case 'KnockDown':
Modifiers[7] = 1.f + (Value*Progress);
break;
2017-10-19 21:00:49 -05:00
case 'Welder':
Modifiers[8] = 1.f + (Value*Progress);
break;
case 'Heal':
Modifiers[9] = 1.f + (Value*Progress);
break;
case 'Mag':
Modifiers[10] = 1.f + (Value*Progress);
2020-11-28 23:12:58 +03:00
if (bActivePerk && WorldInfo.NetMode!=NM_Client && PlayerOwner.Pawn!=None && PlayerOwner.Pawn.InvManager!=None)
2017-10-19 21:00:49 -05:00
UpdateAmmoStatus(PlayerOwner.Pawn.InvManager);
break;
case 'Spare':
Modifiers[11] = 1.f + (Value*Progress);
2020-11-28 23:12:58 +03:00
if (bActivePerk && WorldInfo.NetMode!=NM_Client && PlayerOwner.Pawn!=None && PlayerOwner.Pawn.InvManager!=None)
2017-10-19 21:00:49 -05:00
UpdateAmmoStatus(PlayerOwner.Pawn.InvManager);
break;
case 'OffDamage':
Modifiers[12] = 1.f + (Value*Progress);
break;
case 'SelfDamage':
Modifiers[13] = 1.f / (1.f+Value*Progress);
break;
case 'Armor':
Modifiers[14] = (Value*Progress*100.f);
2020-11-28 23:12:58 +03:00
if (bActivePerk && KFPawn_Human(PlayerOwner.Pawn)!=None)
2017-10-19 21:00:49 -05:00
{
KFPawn_Human(PlayerOwner.Pawn).MaxArmor = KFPawn_Human(PlayerOwner.Pawn).Default.MaxArmor;
ModifyArmor(KFPawn_Human(PlayerOwner.Pawn).MaxArmor);
}
return FMin(Value*Progress,1.55);
case 'PoisonDmg':
Modifiers[15] = 1.f / (1.f+Value*Progress);
break;
case 'SonicDmg':
Modifiers[16] = 1.f / (1.f+Value*Progress);
break;
case 'FireDmg':
Modifiers[17] = 1.f / (1.f+Value*Progress);
break;
case 'AllDmg':
Modifiers[18] = 1.f / (1.f+Value*Progress);
break;
case 'HeadDamage':
Modifiers[19] = Value*Progress;
break;
case 'HealRecharge':
Modifiers[20] = 1.f / (1.f+Value*Progress);
break;
2021-08-21 10:53:27 -07:00
case 'Switch':
Modifiers[21] = 1.f / (1.f+Value*Progress);
2023-05-14 05:49:12 +03:00
break;
2017-10-19 21:00:49 -05:00
}
return (Value*Progress);
}
2020-11-28 23:04:55 +03:00
simulated function ModifyDamageGiven(out int InDamage, optional Actor DamageCauser, optional KFPawn_Monster MyKFPM, optional KFPlayerController DamageInstigator, optional class<KFDamageType> DamageType, optional int HitZoneIdx)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (BasePerk==None || (DamageType!=None && DamageType.Default.ModifierPerkList.Find(BasePerk)>=0) || (KFWeapon(DamageCauser)!=None && IsWeaponOnPerk(KFWeapon(DamageCauser))))
{
2020-11-28 23:12:58 +03:00
if (HitZoneIdx == 0)
InDamage *= (Modifiers[1] + Modifiers[19]);
else
InDamage *= Modifiers[1];
}
2020-11-28 23:12:58 +03:00
else if (DamageType==None || DamageType.Name!='KFDT_SuicideExplosive')
2017-10-19 21:00:49 -05:00
InDamage *= Modifiers[12];
}
2020-11-29 00:54:57 +03:00
2020-11-28 23:04:55 +03:00
simulated function ModifyDamageTaken(out int InDamage, optional class<DamageType> DamageType, optional Controller InstigatedBy)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (InDamage>0)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if ((InstigatedBy==None || InstigatedBy==PlayerOwner) && class<KFDamageType>(DamageType)!=None)
2017-10-19 21:00:49 -05:00
InDamage *= Modifiers[13];
2020-11-28 23:12:58 +03:00
else if (Modifiers[15]<1 && class<KFDT_Toxic>(DamageType)!=None)
2017-10-19 21:00:49 -05:00
InDamage = Max(InDamage*Modifiers[15],1); // Do at least 1 damage.
2020-11-28 23:12:58 +03:00
else if (Modifiers[16]<1 && class<KFDT_Sonic>(DamageType)!=None)
2017-10-19 21:00:49 -05:00
InDamage = Max(InDamage*Modifiers[16],1);
2020-11-28 23:12:58 +03:00
else if (Modifiers[17]<1 && class<KFDT_Fire>(DamageType)!=None)
2017-10-19 21:00:49 -05:00
InDamage = Max(InDamage*Modifiers[17],1);
2020-11-28 23:12:58 +03:00
if (Modifiers[18]<1 && InstigatedBy!=None && InstigatedBy!=PlayerOwner)
2017-10-19 21:00:49 -05:00
InDamage = Max(InDamage*Modifiers[18],1);
}
}
2020-11-29 00:54:57 +03:00
2020-11-28 23:04:55 +03:00
simulated function ModifyRecoil(out float CurrentRecoilModifier, KFWeapon KFW)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (IsWeaponOnPerk(KFW))
2017-10-19 21:00:49 -05:00
CurrentRecoilModifier *= Modifiers[2];
}
2020-11-29 00:54:57 +03:00
2020-11-28 23:04:55 +03:00
simulated function ModifySpread(out float InSpread)
2017-10-19 21:00:49 -05:00
{
InSpread *= Modifiers[3];
}
2020-11-29 00:54:57 +03:00
2020-11-28 23:04:55 +03:00
simulated function ModifyRateOfFire(out float InRate, KFWeapon KFW)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (IsWeaponOnPerk(KFW))
2017-10-19 21:00:49 -05:00
InRate *= Modifiers[4];
}
2020-11-29 00:54:57 +03:00
2017-10-19 21:00:49 -05:00
simulated function float GetReloadRateScale(KFWeapon KFW)
{
return (IsWeaponOnPerk(KFW) ? Modifiers[5] : 1.f);
}
2020-11-29 00:54:57 +03:00
2021-08-21 07:25:27 -07:00
simulated function float GetCameraViewShakeModifier(KFWeapon KFW)
2023-05-14 05:49:12 +03:00
{
2021-08-24 02:03:16 +03:00
return Modifiers[2];
2021-08-21 07:25:27 -07:00
}
2020-11-28 23:04:55 +03:00
function ModifyHealth(out int InHealth)
2017-10-19 21:00:49 -05:00
{
InHealth *= Modifiers[6];
}
2020-11-29 00:54:57 +03:00
2020-11-28 23:04:55 +03:00
function ModifyArmor(out byte MaxArmor)
2017-10-19 21:00:49 -05:00
{
MaxArmor = Min(MaxArmor+Modifiers[14],255);
}
2020-11-29 00:54:57 +03:00
2017-10-19 21:00:49 -05:00
function float GetKnockdownPowerModifier()
{
return Modifiers[7];
}
2020-11-29 00:54:57 +03:00
2020-11-28 23:04:55 +03:00
function float GetStunPowerModifier(optional class<DamageType> DamageType, optional byte HitZoneIdx)
2017-10-19 21:00:49 -05:00
{
return Modifiers[7];
}
function float GetStumblePowerModifier( optional KFPawn KFP, optional class<KFDamageType> DamageType, optional out float CooldownModifier, optional byte BodyPart )
{
2021-08-24 02:25:27 +03:00
return Modifiers[7];
}
2020-11-28 23:04:55 +03:00
simulated function ModifyMeleeAttackSpeed(out float InDuration);
2017-10-19 21:00:49 -05:00
2020-11-28 23:04:55 +03:00
function AddDefaultInventory(KFPawn P)
2017-10-19 21:00:49 -05:00
{
local int i;
2020-11-28 23:12:58 +03:00
if (PrimaryWeapon!=None)
2017-10-19 21:00:49 -05:00
P.DefaultInventory.AddItem(PrimaryWeapon);
P.DefaultInventory.AddItem(PrimaryMelee);
2020-11-28 23:12:58 +03:00
if (KFInventoryManager(P.InvManager)!=None)
2017-10-19 21:00:49 -05:00
KFInventoryManager(P.InvManager).MaxCarryBlocks = KFInventoryManager(P.InvManager).Default.MaxCarryBlocks+Modifiers[10];
2023-05-14 05:49:12 +03:00
2020-11-28 23:12:58 +03:00
for (i=0; i<PerkTraits.Length; ++i)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (PerkTraits[i].CurrentLevel>0)
2017-10-19 21:00:49 -05:00
PerkTraits[i].TraitType.Static.AddDefaultInventory(P,Self,PerkTraits[i].CurrentLevel,PerkTraits[i].Data);
}
}
2020-11-28 23:04:55 +03:00
simulated function ModifyWeldingRate(out float FastenRate, out float UnfastenRate)
2017-10-19 21:00:49 -05:00
{
FastenRate *= Modifiers[8];
UnfastenRate *= Modifiers[8];
}
2020-11-28 23:04:55 +03:00
function bool RepairArmor(Pawn HealTarget)
2017-10-19 21:00:49 -05:00
{
return false;
}
2020-11-29 00:54:57 +03:00
2020-11-28 23:04:55 +03:00
function bool ModifyHealAmount(out float HealAmount)
2017-10-19 21:00:49 -05:00
{
HealAmount*=Modifiers[9];
return false;
}
2020-11-29 00:54:57 +03:00
2020-11-28 23:04:55 +03:00
simulated function ModifyMagSizeAndNumber(KFWeapon KFW, out int MagazineCapacity, optional array< Class<KFPerk> > WeaponPerkClass, optional bool bSecondary=false, optional name WeaponClassname)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (MagazineCapacity>2 && (KFW==None ? WeaponPerkClass.Find(BasePerk)>=0 : IsWeaponOnPerk(KFW))) // Skip boomstick for this.
2017-10-19 21:00:49 -05:00
MagazineCapacity = Min(MagazineCapacity*Modifiers[10],255);
}
2020-11-29 00:54:57 +03:00
2020-11-28 23:04:55 +03:00
simulated function ModifySpareAmmoAmount(KFWeapon KFW, out int PrimarySpareAmmo, optional const out STraderItem TraderItem, optional bool bSecondary)
2017-10-19 21:00:49 -05:00
{
2020-11-28 23:12:58 +03:00
if (KFW==None ? TraderItem.AssociatedPerkClasses.Find(BasePerk)>=0 : IsWeaponOnPerk(KFW))
2017-10-19 21:00:49 -05:00
PrimarySpareAmmo*=Modifiers[11];
}
2020-11-29 00:54:57 +03:00
2020-11-28 23:04:55 +03:00
simulated function bool ShouldMagSizeModifySpareAmmo(KFWeapon KFW, optional Class<KFPerk> WeaponPerkClass)
2017-10-19 21:00:49 -05:00
{
return (KFW==None ? WeaponPerkClass==BasePerk : IsWeaponOnPerk(KFW));
}
2020-11-28 23:04:55 +03:00
final function UpdateAmmoStatus(InventoryManager Inv)
2017-10-19 21:00:49 -05:00
{
local KFWeapon W;
foreach Inv.InventoryActors(class'KFWeapon',W)
{
// if (IsWeaponOnPerk(W))
W.ReInitializeAmmoCounts(PerkManager);
2017-10-19 21:00:49 -05:00
}
}
2020-11-28 23:04:55 +03:00
simulated function ModifyHealerRechargeTime(out float RechargeRate)
{
RechargeRate *= Modifiers[20];
}
2017-10-19 21:00:49 -05:00
simulated function DrawSpecialPerkHUD(Canvas C)
{
2020-11-28 23:12:58 +03:00
if (EnemyHealthRange>0 && PlayerOwner!=None && KFPawn_Human(PlayerOwner.Pawn)!=None)
2017-10-19 21:00:49 -05:00
DrawEnemyHealth(C);
}
2020-11-28 23:04:55 +03:00
simulated final function DrawEnemyHealth(Canvas C)
2017-10-19 21:00:49 -05:00
{
local KFPawn_Monster KFPM;
local vector X,CameraLocation;
2023-05-14 05:49:12 +03:00
2017-10-19 21:00:49 -05:00
X = vector(PlayerOwner.Pawn.GetViewRotation());
CameraLocation = PlayerOwner.Pawn.GetPawnViewLocation();
foreach WorldInfo.AllPawns(class'KFPawn_Monster',KFPM,CameraLocation,EnemyDistDraw[EnemyHealthRange-1])
{
2020-11-28 23:12:58 +03:00
if (KFPM.IsAliveAndWell() && `TimeSince(KFPM.Mesh.LastRenderTime)<0.1f && KFPM.CanShowHealth() && KFPM.GetTeamNum()!=0 && (X Dot (KFPM.Location - CameraLocation))>0)
2017-10-19 21:00:49 -05:00
DrawZedHealthbar(C,KFPM,CameraLocation);
}
}
2020-11-28 23:04:55 +03:00
simulated final function DrawZedHealthbar(Canvas C, KFPawn_Monster KFPM, vector CameraLocation)
2017-10-19 21:00:49 -05:00
{
local vector ScreenPos, TargetLocation;
local float HealthBarLength, HealthbarHeight, HealthScale;
HealthbarLength = FMin(50.f * (float(C.SizeX) / 1024.f), 50.f) * class'KFGFxHudWrapper'.Default.FriendlyHudScale;
HealthbarHeight = FMin(6.f * (float(C.SizeX) / 1024.f), 6.f) * class'KFGFxHudWrapper'.Default.FriendlyHudScale;
HealthScale = FClamp(float(KFPM.Health) / float(KFPM.HealthMax),0.f,1.f);
2020-11-28 23:12:58 +03:00
if (KFPM.bCrawler && KFPM.Floor.Z <= -0.7f && KFPM.Physics == PHYS_Spider)
2017-10-19 21:00:49 -05:00
{
TargetLocation = KFPM.Location + vect(0,0,-1) * KFPM.GetCollisionHeight() * 1.2;
}
else
{
TargetLocation = KFPM.Location + vect(0,0,1) * KFPM.GetCollisionHeight() * 1.2;
}
ScreenPos = C.Project(TargetLocation);
2020-11-28 23:12:58 +03:00
if (ScreenPos.X < 0 || ScreenPos.X > C.SizeX || ScreenPos.Y < 0 || ScreenPos.Y > C.SizeY || !FastTrace(TargetLocation, CameraLocation))
2017-10-19 21:00:49 -05:00
return;
C.EnableStencilTest(true);
C.SetDrawColor(0, 0, 0, 255);
C.SetPos(ScreenPos.X - HealthBarLength * 0.5, ScreenPos.Y);
C.DrawTileStretched(class'KFPerk_Commando'.Default.WhiteMaterial, HealthbarLength, HealthbarHeight, 0, 0, 32, 32);
C.SetDrawColor(237, 8, 0, 255);
C.SetPos(ScreenPos.X - HealthBarLength * 0.5 + 1.0, ScreenPos.Y + 1.0);
C.DrawTileStretched(class'KFPerk_Commando'.Default.WhiteMaterial, (HealthBarLength - 2.0) * HealthScale, HealthbarHeight - 2.0, 0, 0, 32, 32);
C.EnableStencilTest(false);
}
2020-11-28 23:04:55 +03:00
function PlayerKilled(KFPawn_Monster Victim, class<DamageType> DamageType);
2017-10-19 21:00:49 -05:00
2020-11-28 23:04:55 +03:00
function ModifyBloatBileDoT(out float DoTScaler)
2017-10-19 21:00:49 -05:00
{
DoTScaler = Modifiers[15];
}
2020-11-28 23:04:55 +03:00
simulated function bool GetIsUberAmmoActive(KFWeapon KFW)
2017-10-19 21:00:49 -05:00
{
return false;
}
2020-11-28 23:04:55 +03:00
function UpdatePerkHeadShots(ImpactInfo Impact, class<DamageType> DamageType, int NumHit);
2017-10-19 21:00:49 -05:00
2020-11-28 23:04:55 +03:00
function CheckForAirborneAgent(KFPawn HealTarget, class<DamageType> DamType, int HealAmount);
2017-10-19 21:00:49 -05:00
2020-11-28 23:04:55 +03:00
simulated function float GetZedTimeModifier(KFWeapon W)
2017-10-19 21:00:49 -05:00
{
return 0.f;
}
2020-11-28 23:04:55 +03:00
simulated function bool GetUsingTactialReload(KFWeapon KFW)
2017-10-19 21:00:49 -05:00
{
return (bTacticalReload && IsWeaponOnPerk(KFW));
}
2020-11-28 23:04:55 +03:00
simulated function float GetIronSightSpeedModifier(KFWeapon KFW)
2017-10-19 21:00:49 -05:00
{
return 1.f;
}
2021-08-24 03:18:38 +03:00
simulated function ModifyWeaponSwitchTime(out float ModifiedSwitchTime)
2021-08-21 10:53:27 -07:00
{
2021-08-24 03:18:38 +03:00
ModifiedSwitchTime *= Modifiers[21];
2021-08-21 10:53:27 -07:00
}
function OnWaveEnded();
function NotifyZedTimeStarted();
2020-11-28 23:04:55 +03:00
simulated function float GetZedTimeExtensions(byte Level)
{
return 1.f;
}
2021-08-21 09:56:06 -07:00
simulated function float GetTightChokeModifier()
{
2023-05-14 06:01:17 +03:00
return Modifiers[3];
2021-08-21 09:56:06 -07:00
}
2017-10-19 21:00:49 -05:00
defaultproperties
{
CurrentConfigVer=15
2017-10-19 21:00:49 -05:00
bOnlyRelevantToOwner=true
bCanBeGrabbed=true
NetUpdateFrequency=1
GrenadeClass=class'KFProj_FragGrenade'
PerkGrenade=class'KFProj_FragGrenade'
SuperGrenade=class'ExtProj_SUPERGrenade'
HealExpUpNum=12
WeldExpUpNum=180
ToxicDartDamage=15
NetPriority=4
2023-05-14 05:49:12 +03:00
2017-10-19 21:00:49 -05:00
SecondaryWeaponDef=class'KFWeapDef_9mm'
KnifeWeaponDef=class'KFWeapDef_Knife_Commando'
GrenadeWeaponDef=class'KFWeapDef_Grenade_Support'
2023-05-14 05:49:12 +03:00
2017-10-19 21:00:49 -05:00
DefTraitList.Add(class'Ext_TraitGrenadeUpg')
DefTraitList.Add(class'Ext_TraitNightvision')
DefTraitList.Add(class'Ext_TraitAmmoReg')
DefTraitList.Add(class'Ext_TraitHealthReg')
DefTraitList.Add(class'Ext_TraitArmorReg')
DefTraitList.Add(class'Ext_TraitCarryCap')
DefTraitList.Add(class'Ext_TraitGrenadeCap')
DefTraitList.Add(class'Ext_TraitMedicPistol')
DefTraitList.Add(class'Ext_TraitZED_Summon')
DefTraitList.Add(class'Ext_TraitZED_Health')
DefTraitList.Add(class'Ext_TraitZED_Damage')
DefTraitList.Add(class'Ext_TraitZED_SummonExt')
DefTraitList.Add(class'Ext_TraitGhost')
DefTraitList.Add(class'Ext_TraitRetali')
DefTraitList.Add(class'Ext_TraitDuracell')
DefTraitList.Add(class'Ext_TraitRagdoll')
DefTraitList.Add(class'Ext_TraitAutoFire')
DefTraitList.Add(class'Ext_TraitBunnyHop')
DefTraitList.Add(class'Ext_TraitKnockback')
WebConfigs.Add((PropType=0,PropName="FirstLevelExp",UIName="First Level XP",UIDesc="EXP required for the FIRST level"))
WebConfigs.Add((PropType=0,PropName="LevelUpExpCost",UIName="Level Up XP",UIDesc="EXP cost for every next level (Level * ThisValue"))
WebConfigs.Add((PropType=0,PropName="LevelUpIncCost",UIName="Level Up Inc XP",UIDesc="Increased EXP cost for every level ((Level^2) * ThisValue)"))
WebConfigs.Add((PropType=0,PropName="MinimumLevel",UIName="Minimum Level",UIDesc="The minimum level of players"))
WebConfigs.Add((PropType=0,PropName="MaximumLevel",UIName="Maximum Level",UIDesc="The maximum level of players"))
WebConfigs.Add((PropType=0,PropName="StarPointsPerLevel",UIName="Star Points Per Lvl",UIDesc="Number of star points players earn per level"))
WebConfigs.Add((PropType=2,PropName="TraitClasses",UIName="Trait Classes",UIDesc="The class names of traits players can buy",NumElements=-1))
WebConfigs.Add((PropType=2,PropName="PerkStats",UIName="Perk Stats",UIDesc="List of perk stats (format in: StatName,Max Stat,Cost Per Stat,Progress Per Level)",NumElements=-1))
WebConfigs.Add((PropType=0,PropName="MinLevelForPrestige",UIName="Min Level For Prestige",UIDesc="Minimum level required to prestige the perk (-1 = disabled)"))
WebConfigs.Add((PropType=0,PropName="PrestigeSPIncrease",UIName="Prestige SP Increase",UIDesc="Star points increase per level for every prestige"))
WebConfigs.Add((PropType=0,PropName="MaxPrestige",UIName="Max Prestige",UIDesc="Maximum prestige level"))
WebConfigs.Add((PropType=0,PropName="PrestigeXPReduce",UIName="Prestige XP Reduce",UIDesc="Percent amount of XP cost is reduced for each prestige (1.0 = 1/2, or 50 % of XP)"))
// WebConfigs.Add((PropType=0,PropName="MinimalDataLevel",UIName="Minimal Real Level",UIDesc="Minimal level for new players or who loads from saves"))
2023-05-14 05:49:12 +03:00
2020-09-01 11:07:53 +03:00
DefPerkStats(0)=(MaxValue=50,CostPerValue=1,StatType="Speed",Progress=0.4)
DefPerkStats(1)=(MaxValue=1000,CostPerValue=1,StatType="Damage",Progress=0.5)
2023-07-17 19:11:20 -07:00
DefPerkStats(2)=(MaxValue=100,CostPerValue=1,StatType="Recoil",Progress=1)
DefPerkStats(3)=(MaxValue=100,CostPerValue=1,StatType="Spread",Progress=1)
2020-09-01 11:07:53 +03:00
DefPerkStats(4)=(MaxValue=1000,CostPerValue=1,StatType="Rate",Progress=0.5)
DefPerkStats(5)=(MaxValue=1000,CostPerValue=1,StatType="Reload",Progress=0.5)
DefPerkStats(6)=(MaxValue=150,CostPerValue=1,StatType="Health",Progress=1)
DefPerkStats(7)=(MaxValue=100,CostPerValue=1,StatType="KnockDown",Progress=1)
DefPerkStats(8)=(MaxValue=200,CostPerValue=1,StatType="Welder",bHiddenConfig=true,Progress=0.5)
DefPerkStats(9)=(MaxValue=400,CostPerValue=1,StatType="Heal",bHiddenConfig=true,Progress=0.5)
DefPerkStats(10)=(MaxValue=400,CostPerValue=1,StatType="Mag",Progress=1)
DefPerkStats(11)=(MaxValue=500,CostPerValue=1,StatType="Spare",Progress=1)
DefPerkStats(12)=(MaxValue=1000,CostPerValue=1,StatType="OffDamage",Progress=0.25)
DefPerkStats(13)=(MaxValue=1000,CostPerValue=1,StatType="SelfDamage",Progress=1,bHiddenConfig=true)
DefPerkStats(14)=(MaxValue=150,CostPerValue=1,StatType="Armor",Progress=1)
DefPerkStats(15)=(MaxValue=1000,CostPerValue=1,StatType="PoisonDmg",Progress=1.5,bHiddenConfig=true)
DefPerkStats(16)=(MaxValue=1000,CostPerValue=1,StatType="SonicDmg",Progress=1.5,bHiddenConfig=true)
DefPerkStats(17)=(MaxValue=1000,CostPerValue=1,StatType="FireDmg",Progress=1.5,bHiddenConfig=true)
DefPerkStats(18)=(MaxValue=500,CostPerValue=1,StatType="AllDmg",Progress=0.25)
DefPerkStats(19)=(MaxValue=500,CostPerValue=1,StatType="HeadDamage",Progress=1,bHiddenConfig=true)
DefPerkStats(20)=(MaxValue=200,CostPerValue=1,StatType="HealRecharge",Progress=0.5,bHiddenConfig=true)
2021-08-21 10:53:27 -07:00
DefPerkStats(21)=(MaxValue=100,CostPerValue=1,StatType="Switch",Progress=1)
2017-10-19 21:00:49 -05:00
Modifiers.Add(1.f)
Modifiers.Add(1.f)
Modifiers.Add(1.f)
Modifiers.Add(1.f)
Modifiers.Add(1.f)
Modifiers.Add(1.f)
Modifiers.Add(1.f)
Modifiers.Add(1.f)
Modifiers.Add(1.f)
Modifiers.Add(1.f)
Modifiers.Add(1.f)
Modifiers.Add(1.f)
Modifiers.Add(1.f)
Modifiers.Add(1.f)
Modifiers.Add(0.f)
Modifiers.Add(1.f)
Modifiers.Add(1.f)
Modifiers.Add(1.f)
Modifiers.Add(1.f)
Modifiers.Add(0.f)
Modifiers.Add(1.f)
2021-08-21 10:53:27 -07:00
Modifiers.Add(1.f)
2023-05-14 05:49:12 +03:00
2017-10-19 21:00:49 -05:00
EnemyDistDraw.Add(500)
EnemyDistDraw.Add(700)
EnemyDistDraw.Add(1000)
EnemyDistDraw.Add(1600)
2021-08-24 02:03:16 +03:00
}