2023-05-13 02:33:32 +03:00
|
|
|
class Unlocker extends Object
|
2023-10-02 01:55:09 +03:00
|
|
|
dependson(WeaponReplacements)
|
2023-05-13 02:33:32 +03:00
|
|
|
abstract;
|
|
|
|
|
|
|
|
// TODO:
|
|
|
|
// replace shopContainer (KFGFxTraderContainer_Store)
|
|
|
|
// without replacing KFGFxMoviePlayer_Manager
|
|
|
|
// but how? 🤔
|
|
|
|
|
2023-10-02 01:55:09 +03:00
|
|
|
const Trader = class'Trader';
|
|
|
|
const Replacements = class'WeaponReplacements';
|
2023-05-13 02:33:32 +03:00
|
|
|
|
|
|
|
public static function bool IsValidTypeUnlockDLC(String UnlockType, E_LogLevel LogLevel)
|
|
|
|
{
|
|
|
|
`Log_TraceStatic();
|
|
|
|
|
|
|
|
switch (Locs(UnlockType))
|
|
|
|
{
|
|
|
|
case "true":
|
|
|
|
case "false":
|
|
|
|
case "auto":
|
|
|
|
case "replaceweapons":
|
|
|
|
case "replacefilter":
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static function bool UnlockDLC(
|
|
|
|
KFGameInfo KFGI,
|
|
|
|
KFGameReplicationInfo KFGRI,
|
|
|
|
String UnlockType,
|
2023-10-20 22:53:11 +03:00
|
|
|
out Array<class<KFWeaponDefinition> > WeapDefs,
|
2023-10-02 01:55:09 +03:00
|
|
|
out BoolWrapper DLCSkinUpdateRequired,
|
2023-05-13 02:33:32 +03:00
|
|
|
E_LogLevel LogLevel)
|
|
|
|
{
|
|
|
|
`Log_TraceStatic();
|
|
|
|
|
|
|
|
switch (Locs(UnlockType))
|
|
|
|
{
|
|
|
|
case "true":
|
|
|
|
case "auto":
|
2023-10-20 22:53:11 +03:00
|
|
|
return Auto(KFGI, KFGRI, WeapDefs, DLCSkinUpdateRequired, LogLevel);
|
2023-05-13 02:33:32 +03:00
|
|
|
|
|
|
|
case "replaceweapons":
|
2023-10-20 22:53:11 +03:00
|
|
|
return ReplaceWeapons(KFGRI, WeapDefs, DLCSkinUpdateRequired, LogLevel);
|
2023-05-13 02:33:32 +03:00
|
|
|
|
|
|
|
case "replacefilter":
|
2023-10-02 01:55:09 +03:00
|
|
|
DLCSkinUpdateRequired.Value = false;
|
2023-05-13 02:33:32 +03:00
|
|
|
return ReplaceFilter(KFGI, LogLevel);
|
|
|
|
|
|
|
|
case "false":
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static function bool Auto(
|
|
|
|
KFGameInfo KFGI,
|
|
|
|
KFGameReplicationInfo KFGRI,
|
2023-10-20 22:53:11 +03:00
|
|
|
out Array<class<KFWeaponDefinition> > WeapDefs,
|
2023-10-02 01:55:09 +03:00
|
|
|
out BoolWrapper DLCSkinUpdateRequired,
|
2023-05-13 02:33:32 +03:00
|
|
|
E_LogLevel LogLevel)
|
|
|
|
{
|
|
|
|
`Log_TraceStatic();
|
|
|
|
|
|
|
|
if (KFGI == None) return false;
|
|
|
|
|
2024-01-04 01:45:36 +03:00
|
|
|
if (CustomGFxManager(KFGI))
|
2023-05-13 02:33:32 +03:00
|
|
|
{
|
2024-01-04 01:45:36 +03:00
|
|
|
return ReplaceWeapons(KFGRI, WeapDefs, DLCSkinUpdateRequired, LogLevel);
|
2023-05-13 02:33:32 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2024-01-04 01:45:36 +03:00
|
|
|
DLCSkinUpdateRequired.Value = false;
|
|
|
|
return ReplaceFilter(KFGI, LogLevel);
|
2023-05-13 02:33:32 +03:00
|
|
|
}
|
2024-01-04 01:45:36 +03:00
|
|
|
}
|
2023-05-13 02:33:32 +03:00
|
|
|
|
2024-01-04 01:45:36 +03:00
|
|
|
public static function bool CustomGFxManager(KFGameInfo KFGI)
|
|
|
|
{
|
|
|
|
if (KFGameInfo_VersusSurvival(KFGI) != None)
|
2023-05-13 02:33:32 +03:00
|
|
|
{
|
2024-01-04 01:45:36 +03:00
|
|
|
return (KFGI.KFGFxManagerClass != class'KFGameInfo_VersusSurvival'.default.KFGFxManagerClass);
|
2023-05-13 02:33:32 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2024-01-04 01:45:36 +03:00
|
|
|
return (KFGI.KFGFxManagerClass != class'KFGameInfo'.default.KFGFxManagerClass);
|
2023-05-13 02:33:32 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static function bool ReplaceWeapons(
|
|
|
|
KFGameReplicationInfo KFGRI,
|
2023-10-20 22:53:11 +03:00
|
|
|
out Array<class<KFWeaponDefinition> > WeapDefs,
|
|
|
|
out BoolWrapper DLCSkinUpdateRequired,
|
2023-05-13 02:33:32 +03:00
|
|
|
E_LogLevel LogLevel)
|
|
|
|
{
|
2023-10-20 22:53:11 +03:00
|
|
|
local class<KFWeaponDefinition> WeapDef;
|
2023-05-13 02:33:32 +03:00
|
|
|
local class<KFWeaponDefinition> WeapDefReplacement;
|
|
|
|
local bool Unlock, PartialUnlock;
|
2023-10-20 22:53:11 +03:00
|
|
|
local int Index;
|
2023-05-13 02:33:32 +03:00
|
|
|
|
|
|
|
`Log_TraceStatic();
|
|
|
|
|
2023-10-02 01:55:09 +03:00
|
|
|
`Log_Debug("Unlock by replace weapons");
|
|
|
|
|
2023-05-13 02:33:32 +03:00
|
|
|
Unlock = false;
|
|
|
|
PartialUnlock = false;
|
2023-10-20 22:53:11 +03:00
|
|
|
DLCSkinUpdateRequired.Value = false;
|
2023-05-13 02:33:32 +03:00
|
|
|
|
2023-10-20 22:53:11 +03:00
|
|
|
for (Index = 0; Index < WeapDefs.Length; Index++)
|
2023-05-13 02:33:32 +03:00
|
|
|
{
|
2023-10-20 22:53:11 +03:00
|
|
|
WeapDef = WeapDefs[Index];
|
|
|
|
|
|
|
|
if (WeapDef.default.SharedUnlockId == SCU_None) continue;
|
|
|
|
|
|
|
|
WeapDefReplacement = PickReplacementWeapDefDLC(WeapDef, LogLevel);
|
2023-05-13 02:33:32 +03:00
|
|
|
if (WeapDefReplacement != None)
|
|
|
|
{
|
|
|
|
Unlock = true;
|
2023-10-20 22:53:11 +03:00
|
|
|
DLCSkinUpdateRequired.Value = true;
|
|
|
|
if (WeapDefs.Find(WeapDefReplacement) == INDEX_NONE)
|
2023-05-13 02:33:32 +03:00
|
|
|
{
|
2023-10-20 22:53:11 +03:00
|
|
|
WeapDefs[Index] = WeapDefReplacement;
|
|
|
|
`Log_Debug(WeapDef @ "replaced by" @ WeapDefReplacement);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
WeapDefs.Remove(Index--, 1);
|
|
|
|
`Log_Debug("Skip already unlocked weapon:" @ WeapDef);
|
2023-05-13 02:33:32 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
PartialUnlock = true;
|
2023-10-20 22:53:11 +03:00
|
|
|
`Log_Warn("Can't unlock item:" @ WeapDef @ "SharedUnlockId:" @ WeapDef.default.SharedUnlockId);
|
2023-09-11 03:26:53 +03:00
|
|
|
}
|
2023-05-13 02:33:32 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if (PartialUnlock)
|
|
|
|
{
|
|
|
|
`Log_Warn("Some DLCs are not unlocked. Try to set 'UnlockDLC=ReplaceFilter' or ask the author to update the mod");
|
|
|
|
}
|
|
|
|
|
|
|
|
return Unlock;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static function class<KFWeaponDefinition> PickReplacementWeapDefDLC(class<KFWeaponDefinition> WeapDefDLC, E_LogLevel LogLevel)
|
|
|
|
{
|
2023-10-02 01:55:09 +03:00
|
|
|
local SWeapReplace WeapReplace;
|
2023-05-13 02:33:32 +03:00
|
|
|
|
|
|
|
`Log_TraceStatic();
|
|
|
|
|
2023-10-02 01:55:09 +03:00
|
|
|
foreach Replacements.default.DLC(WeapReplace)
|
2023-05-13 02:33:32 +03:00
|
|
|
{
|
2023-10-02 01:55:09 +03:00
|
|
|
if (ClassIsChildOf(WeapReplace.WeapDef, WeapDefDLC))
|
2023-05-13 02:33:32 +03:00
|
|
|
{
|
2023-10-02 01:55:09 +03:00
|
|
|
return WeapReplace.WeapDef;
|
2023-05-13 02:33:32 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static function bool ReplaceFilter(KFGameInfo KFGI, E_LogLevel LogLevel)
|
|
|
|
{
|
|
|
|
`Log_TraceStatic();
|
|
|
|
|
2023-10-02 01:55:09 +03:00
|
|
|
`Log_Debug("Unlock by replace filter");
|
|
|
|
|
2023-05-13 02:33:32 +03:00
|
|
|
if (KFGI == None) return false;
|
|
|
|
|
2024-01-04 01:45:36 +03:00
|
|
|
if (CustomGFxManager(KFGI))
|
|
|
|
{
|
|
|
|
`Log_Warn("Custom KFGFxMoviePlayer_Manager detected:" @ String(KFGI.KFGFxManagerClass) $ ". There may be compatibility issues.");
|
|
|
|
}
|
|
|
|
|
2023-05-13 02:33:32 +03:00
|
|
|
if (KFGameInfo_VersusSurvival(KFGI) != None)
|
|
|
|
{
|
2024-01-04 01:45:36 +03:00
|
|
|
KFGI.KFGFxManagerClass = class'CTI_GFxMoviePlayer_Manager_Versus_DLC';
|
2023-05-13 02:33:32 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2024-01-04 01:45:36 +03:00
|
|
|
KFGI.KFGFxManagerClass = class'CTI_GFxMoviePlayer_Manager_DLC';
|
2023-05-13 02:33:32 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
defaultproperties
|
|
|
|
{
|
2023-10-02 01:55:09 +03:00
|
|
|
|
2023-05-13 02:33:32 +03:00
|
|
|
}
|