2023-03-03 15:23:46 +00:00
import { inject , injectable } from "tsyringe" ;
2023-10-19 17:21:17 +00:00
import { BotGeneratorHelper , ExhaustableArray } from "@spt-aki/helpers/BotGeneratorHelper" ;
import { BotHelper } from "@spt-aki/helpers/BotHelper" ;
import { BotWeaponGeneratorHelper } from "@spt-aki/helpers/BotWeaponGeneratorHelper" ;
import { ItemHelper } from "@spt-aki/helpers/ItemHelper" ;
2024-01-12 17:00:22 +00:00
import { PresetHelper } from "@spt-aki/helpers/PresetHelper" ;
2023-10-19 17:21:17 +00:00
import { ProbabilityHelper } from "@spt-aki/helpers/ProbabilityHelper" ;
import { ProfileHelper } from "@spt-aki/helpers/ProfileHelper" ;
2024-01-07 15:34:59 +00:00
import { WeightedRandomHelper } from "@spt-aki/helpers/WeightedRandomHelper" ;
2024-02-05 15:36:05 +00:00
import { IPreset } from "@spt-aki/models/eft/common/IGlobals" ;
2023-10-19 17:21:17 +00:00
import { Mods , ModsChances } from "@spt-aki/models/eft/common/tables/IBotType" ;
import { Item } from "@spt-aki/models/eft/common/tables/IItem" ;
import { ITemplateItem , Slot } from "@spt-aki/models/eft/common/tables/ITemplateItem" ;
import { BaseClasses } from "@spt-aki/models/enums/BaseClasses" ;
import { ConfigTypes } from "@spt-aki/models/enums/ConfigTypes" ;
2024-01-25 13:45:42 +00:00
import { ModSpawn } from "@spt-aki/models/enums/ModSpawn" ;
2024-01-27 18:12:13 +00:00
import { IChooseRandomCompatibleModResult } from "@spt-aki/models/spt/bots/IChooseRandomCompatibleModResult" ;
2024-02-05 22:02:03 +00:00
import { EquipmentFilterDetails , EquipmentFilters , IBotConfig } from "@spt-aki/models/spt/config/IBotConfig" ;
2023-10-19 17:21:17 +00:00
import { ILogger } from "@spt-aki/models/spt/utils/ILogger" ;
import { ConfigServer } from "@spt-aki/servers/ConfigServer" ;
import { DatabaseServer } from "@spt-aki/servers/DatabaseServer" ;
import { BotEquipmentFilterService } from "@spt-aki/services/BotEquipmentFilterService" ;
import { BotEquipmentModPoolService } from "@spt-aki/services/BotEquipmentModPoolService" ;
import { BotModLimits , BotWeaponModLimitService } from "@spt-aki/services/BotWeaponModLimitService" ;
import { ItemFilterService } from "@spt-aki/services/ItemFilterService" ;
import { LocalisationService } from "@spt-aki/services/LocalisationService" ;
import { HashUtil } from "@spt-aki/utils/HashUtil" ;
import { JsonUtil } from "@spt-aki/utils/JsonUtil" ;
import { RandomUtil } from "@spt-aki/utils/RandomUtil" ;
2024-01-07 14:46:25 +00:00
import { IGenerateEquipmentProperties } from "./BotInventoryGenerator" ;
2024-01-12 17:00:22 +00:00
import { IFilterPlateModsForSlotByLevelResult , Result } from "./IFilterPlateModsForSlotByLevelResult" ;
2023-03-03 15:23:46 +00:00
@injectable ( )
export class BotEquipmentModGenerator
{
protected botConfig : IBotConfig ;
constructor (
@inject ( "WinstonLogger" ) protected logger : ILogger ,
@inject ( "JsonUtil" ) protected jsonUtil : JsonUtil ,
@inject ( "HashUtil" ) protected hashUtil : HashUtil ,
@inject ( "RandomUtil" ) protected randomUtil : RandomUtil ,
@inject ( "ProbabilityHelper" ) protected probabilityHelper : ProbabilityHelper ,
@inject ( "DatabaseServer" ) protected databaseServer : DatabaseServer ,
@inject ( "ItemHelper" ) protected itemHelper : ItemHelper ,
@inject ( "BotEquipmentFilterService" ) protected botEquipmentFilterService : BotEquipmentFilterService ,
@inject ( "ItemFilterService" ) protected itemFilterService : ItemFilterService ,
@inject ( "ProfileHelper" ) protected profileHelper : ProfileHelper ,
@inject ( "BotWeaponModLimitService" ) protected botWeaponModLimitService : BotWeaponModLimitService ,
@inject ( "BotHelper" ) protected botHelper : BotHelper ,
@inject ( "BotGeneratorHelper" ) protected botGeneratorHelper : BotGeneratorHelper ,
@inject ( "BotWeaponGeneratorHelper" ) protected botWeaponGeneratorHelper : BotWeaponGeneratorHelper ,
2024-01-07 15:34:59 +00:00
@inject ( "WeightedRandomHelper" ) protected weightedRandomHelper : WeightedRandomHelper ,
2024-01-12 17:00:22 +00:00
@inject ( "PresetHelper" ) protected presetHelper : PresetHelper ,
2023-03-03 15:23:46 +00:00
@inject ( "LocalisationService" ) protected localisationService : LocalisationService ,
@inject ( "BotEquipmentModPoolService" ) protected botEquipmentModPoolService : BotEquipmentModPoolService ,
2023-11-13 11:05:05 -05:00
@inject ( "ConfigServer" ) protected configServer : ConfigServer ,
2023-03-03 15:23:46 +00:00
)
{
this . botConfig = this . configServer . getConfig ( ConfigTypes . BOT ) ;
}
2023-11-13 11:05:05 -05:00
2023-03-03 15:23:46 +00:00
/ * *
* Check mods are compatible and add to array
* @param equipment Equipment item to add mods to
* @param modPool Mod list to choose frm
* @param parentId parentid of item to add mod to
* @param parentTemplate template objet of item to add mods to
* @param forceSpawn should this mod be forced to spawn
* @returns Item + compatible mods as an array
* /
2023-11-13 11:05:05 -05:00
public generateModsForEquipment (
equipment : Item [ ] ,
parentId : string ,
parentTemplate : ITemplateItem ,
2024-01-07 14:46:25 +00:00
settings : IGenerateEquipmentProperties ,
2023-11-13 11:05:05 -05:00
forceSpawn = false ,
) : Item [ ]
2023-03-03 15:23:46 +00:00
{
2024-01-07 14:46:25 +00:00
const compatibleModsPool = settings . modPool [ parentTemplate . _id ] ;
2023-12-28 18:36:37 +00:00
if ( ! compatibleModsPool )
{
2024-02-02 13:54:07 -05:00
this . logger . warning (
` bot: ${ settings . botRole } lacks a mod slot pool for item: ${ parentTemplate . _id } ${ parentTemplate . _name } ` ,
) ;
2023-12-28 18:36:37 +00:00
}
2023-03-03 15:23:46 +00:00
// Iterate over mod pool and choose mods to add to item
2024-01-25 10:55:33 +00:00
for ( const modSlotName in compatibleModsPool )
2023-03-03 15:23:46 +00:00
{
2024-01-25 10:55:33 +00:00
const itemSlotTemplate = this . getModItemSlotFromDb ( modSlotName , parentTemplate ) ;
if ( ! itemSlotTemplate )
2023-03-03 15:23:46 +00:00
{
2023-11-13 11:05:05 -05:00
this . logger . error (
this . localisationService . getText ( "bot-mod_slot_missing_from_item" , {
2024-01-25 10:55:33 +00:00
modSlot : modSlotName ,
2023-11-13 11:05:05 -05:00
parentId : parentTemplate._id ,
parentName : parentTemplate._name ,
2024-02-02 13:54:07 -05:00
botRole : settings.botRole ,
2023-11-13 11:05:05 -05:00
} ) ,
) ;
2023-03-03 15:23:46 +00:00
continue ;
}
2024-02-02 13:54:07 -05:00
const modSpawnResult = this . shouldModBeSpawned (
itemSlotTemplate ,
modSlotName . toLowerCase ( ) ,
settings . spawnChances . equipmentMods ,
) ;
2024-01-25 13:45:42 +00:00
if ( modSpawnResult === ModSpawn . SKIP && ! forceSpawn )
2023-03-03 15:23:46 +00:00
{
continue ;
}
// Ensure submods for nvgs all spawn together
2024-01-25 10:55:33 +00:00
if ( modSlotName === "mod_nvg" )
2023-12-27 23:29:37 +00:00
{
forceSpawn = true ;
}
2023-03-03 15:23:46 +00:00
2024-01-25 10:55:33 +00:00
let modPoolToChooseFrom = compatibleModsPool [ modSlotName ] ;
2024-02-02 13:54:07 -05:00
if (
settings . botEquipmentConfig . filterPlatesByLevel
&& this . itemHelper . isRemovablePlateSlot ( modSlotName . toLowerCase ( ) )
)
2024-01-12 17:00:22 +00:00
{
2024-02-02 13:54:07 -05:00
const outcome = this . filterPlateModsForSlotByLevel (
settings ,
modSlotName . toLowerCase ( ) ,
compatibleModsPool [ modSlotName ] ,
parentTemplate ,
) ;
2024-01-12 17:00:22 +00:00
if ( [ Result . UNKNOWN_FAILURE , Result . NO_DEFAULT_FILTER ] . includes ( outcome . result ) )
{
2024-02-02 13:54:07 -05:00
this . logger . debug (
` Plate slot: ${ modSlotName } selection for armor: ${ parentTemplate . _id } failed: ${
Result [ outcome . result ]
} , skipping ` ,
) ;
2024-01-07 14:46:25 +00:00
2024-01-12 17:00:22 +00:00
continue ;
}
2024-02-02 13:54:07 -05:00
2024-01-12 17:00:22 +00:00
if ( [ Result . LACKS_PLATE_WEIGHTS ] . includes ( outcome . result ) )
{
2024-02-02 13:54:07 -05:00
this . logger . warning (
` Plate slot: ${ modSlotName } lacks weights for armor: ${ parentTemplate . _id } , unable to adjust plate choice, using existing data ` ,
) ;
2024-01-12 17:00:22 +00:00
}
2024-02-02 13:54:07 -05:00
2024-01-12 17:00:22 +00:00
modPoolToChooseFrom = outcome . plateModTpls ;
}
2023-11-13 11:05:05 -05:00
2023-03-03 15:23:46 +00:00
// Find random mod and check its compatible
2024-01-12 17:00:22 +00:00
let modTpl : string ;
let found = false ;
2024-02-02 13:54:07 -05:00
const exhaustableModPool = new ExhaustableArray ( modPoolToChooseFrom , this . randomUtil , this . jsonUtil ) ;
2023-03-03 15:23:46 +00:00
while ( exhaustableModPool . hasValues ( ) )
{
modTpl = exhaustableModPool . getRandomValue ( ) ;
2023-11-13 11:05:05 -05:00
if (
2024-02-02 13:54:07 -05:00
! this . botGeneratorHelper . isItemIncompatibleWithCurrentItems ( equipment , modTpl , modSlotName )
. incompatible
2023-11-13 11:05:05 -05:00
)
2023-03-03 15:23:46 +00:00
{
found = true ;
break ;
}
}
2023-11-13 11:05:05 -05:00
// Compatible item not found but slot REQUIRES item, get random item from db
2024-01-25 10:55:33 +00:00
if ( ! found && itemSlotTemplate . _required )
2023-03-03 15:23:46 +00:00
{
2024-01-25 13:45:42 +00:00
modTpl = this . getRandomModTplFromItemDb ( modTpl , itemSlotTemplate , modSlotName , equipment ) ;
2023-03-03 15:23:46 +00:00
found = ! ! modTpl ;
}
// Compatible item not found + not required
2024-01-25 10:55:33 +00:00
if ( ! ( found || itemSlotTemplate . _required ) )
2023-03-03 15:23:46 +00:00
{
2023-11-13 11:05:05 -05:00
// Don't add item
2023-03-03 15:23:46 +00:00
continue ;
}
const modTemplate = this . itemHelper . getItem ( modTpl ) ;
2024-01-25 10:55:33 +00:00
if ( ! this . isModValidForSlot ( modTemplate , itemSlotTemplate , modSlotName , parentTemplate , settings . botRole ) )
2023-03-03 15:23:46 +00:00
{
continue ;
}
2024-01-25 10:55:33 +00:00
// Generate new id to ensure all items are unique on bot
2023-03-03 15:23:46 +00:00
const modId = this . hashUtil . generate ( ) ;
2024-01-25 10:55:33 +00:00
equipment . push ( this . createModItem ( modId , modTpl , parentId , modSlotName , modTemplate [ 1 ] , settings . botRole ) ) ;
2023-03-03 15:23:46 +00:00
2024-01-12 17:00:22 +00:00
// Does the item being added have possible child mods?
2024-01-07 14:46:25 +00:00
if ( Object . keys ( settings . modPool ) . includes ( modTpl ) )
2023-03-03 15:23:46 +00:00
{
2024-01-12 17:00:22 +00:00
// Call self recursively with item being checkced item we just added to bot
2024-02-02 13:54:07 -05:00
this . generateModsForEquipment ( equipment , modId , modTemplate [ 1 ] , settings , forceSpawn ) ;
2023-03-03 15:23:46 +00:00
}
}
return equipment ;
}
2024-01-07 15:34:59 +00:00
/ * *
* Filter a bots plate pool based on its current level
* @param settings Bot equipment generation settings
2024-01-12 17:00:22 +00:00
* @param modSlot Armor slot being filtered
* @param existingPlateTplPool Plates tpls to choose from
2024-02-02 13:54:07 -05:00
* @param armorItem
2024-01-07 15:34:59 +00:00
* @returns Array of plate tpls to choose from
* /
2024-02-02 13:54:07 -05:00
protected filterPlateModsForSlotByLevel (
settings : IGenerateEquipmentProperties ,
modSlot : string ,
existingPlateTplPool : string [ ] ,
armorItem : ITemplateItem ,
) : IFilterPlateModsForSlotByLevelResult
2024-01-07 14:46:25 +00:00
{
2024-02-02 13:54:07 -05:00
const result : IFilterPlateModsForSlotByLevelResult = { result : Result.UNKNOWN_FAILURE , plateModTpls : null } ;
2024-01-12 17:00:22 +00:00
2024-01-07 14:46:25 +00:00
// Not pmc or not a plate slot, return original mod pool array
2024-01-10 14:47:09 +00:00
if ( ! this . itemHelper . isRemovablePlateSlot ( modSlot ) )
2024-01-07 15:34:59 +00:00
{
2024-01-12 17:00:22 +00:00
result . result = Result . NOT_PLATE_HOLDING_SLOT ;
result . plateModTpls = existingPlateTplPool ;
return result ;
2024-01-07 15:34:59 +00:00
}
// Get the front/back/side weights based on bots level
2024-02-02 13:54:07 -05:00
const plateSlotWeights = settings . botEquipmentConfig ? . armorPlateWeighting ? . find ( ( x ) = >
settings . botLevel >= x . levelRange . min && settings . botLevel <= x . levelRange . max
) ;
2024-01-07 15:34:59 +00:00
if ( ! plateSlotWeights )
{
2024-01-12 17:00:22 +00:00
// No weights, return original array of plate tpls
result . result = Result . LACKS_PLATE_WEIGHTS ;
result . plateModTpls = existingPlateTplPool ;
return result ;
2024-01-07 15:34:59 +00:00
}
// Get the specific plate slot weights (front/back/side)
const plateWeights : Record < string , number > = plateSlotWeights [ modSlot ] ;
if ( ! plateWeights )
2024-01-07 14:46:25 +00:00
{
2024-01-12 17:00:22 +00:00
// No weights, return original array of plate tpls
result . result = Result . LACKS_PLATE_WEIGHTS ;
result . plateModTpls = existingPlateTplPool ;
2024-01-07 15:34:59 +00:00
2024-01-12 17:00:22 +00:00
return result ;
2024-01-07 14:46:25 +00:00
}
2024-01-07 15:34:59 +00:00
// Choose a plate level based on weighting
const chosenArmorPlateLevel = this . weightedRandomHelper . getWeightedValue < string > ( plateWeights ) ;
2024-02-02 13:54:07 -05:00
2024-01-07 15:34:59 +00:00
// Convert the array of ids into database items
2024-02-02 13:54:07 -05:00
const platesFromDb = existingPlateTplPool . map ( ( x ) = > this . itemHelper . getItem ( x ) [ 1 ] ) ;
2024-01-07 15:34:59 +00:00
// Filter plates to the chosen level based on its armorClass property
2024-02-02 13:54:07 -05:00
const filteredPlates = platesFromDb . filter ( ( x ) = > x . _props . armorClass === chosenArmorPlateLevel ) ;
2024-01-09 23:46:57 +00:00
if ( filteredPlates . length === 0 )
2024-01-07 15:34:59 +00:00
{
2024-02-02 13:54:07 -05:00
this . logger . debug (
` Plate filter was too restrictive for armor: ${ armorItem . _id } , unable to find plates of level: ${ chosenArmorPlateLevel } . Using mod items default plate ` ,
) ;
2024-01-07 15:34:59 +00:00
2024-02-02 13:54:07 -05:00
const relatedItemDbModSlot = armorItem . _props . Slots . find ( ( slot ) = > slot . _name . toLowerCase ( ) === modSlot ) ;
const defaultPlate = relatedItemDbModSlot . _props . filters [ 0 ] . Plate ;
2024-01-25 13:45:42 +00:00
if ( ! defaultPlate )
2024-01-12 17:00:22 +00:00
{
// No relevant plate found after filtering AND no default plate
// Last attempt, get default preset and see if it has a plate default
const defaultPreset = this . presetHelper . getDefaultPreset ( armorItem . _id ) ;
if ( defaultPreset )
{
2024-02-02 13:54:07 -05:00
const relatedPresetSlot = defaultPreset . _items . find ( ( item ) = >
item . slotId ? . toLowerCase ( ) === modSlot
) ;
2024-01-12 17:00:22 +00:00
if ( relatedPresetSlot )
{
result . result = Result . SUCCESS ;
result . plateModTpls = [ relatedPresetSlot . _tpl ] ;
return result ;
}
}
result . result = Result . NO_DEFAULT_FILTER ;
return result ;
}
result . result = Result . SUCCESS ;
2024-01-25 13:45:42 +00:00
result . plateModTpls = [ defaultPlate ] ;
2024-01-12 17:00:22 +00:00
return result ;
2024-01-07 15:34:59 +00:00
}
2024-01-07 14:46:25 +00:00
2024-01-07 15:34:59 +00:00
// Only return the items ids
2024-01-12 17:00:22 +00:00
result . result = Result . SUCCESS ;
2024-02-02 13:54:07 -05:00
result . plateModTpls = filteredPlates . map ( ( x ) = > x . _id ) ;
2024-01-07 14:46:25 +00:00
2024-01-12 17:00:22 +00:00
return result ;
2024-01-07 14:46:25 +00:00
}
2023-03-03 15:23:46 +00:00
/ * *
* Add mods to a weapon using the provided mod pool
* @param sessionId session id
* @param weapon Weapon to add mods to
* @param modPool Pool of compatible mods to attach to weapon
2024-01-25 13:45:42 +00:00
* @param weaponId parentId of weapon
2023-03-03 15:23:46 +00:00
* @param parentTemplate Weapon which mods will be generated on
* @param modSpawnChances Mod spawn chances
* @param ammoTpl Ammo tpl to use when generating magazines / cartridges
* @param botRole Role of bot weapon is generated for
2023-11-13 11:05:05 -05:00
* @param botLevel Level of the bot weapon is being generated for
* @param modLimits limits placed on certain mod types per gun
2023-03-03 15:23:46 +00:00
* @param botEquipmentRole role of bot when accessing bot . json equipment config settings
* @returns Weapon + mods array
* /
public generateModsForWeapon (
sessionId : string ,
weapon : Item [ ] ,
modPool : Mods ,
2024-01-25 13:45:42 +00:00
weaponId : string ,
2023-03-03 15:23:46 +00:00
parentTemplate : ITemplateItem ,
modSpawnChances : ModsChances ,
ammoTpl : string ,
botRole : string ,
botLevel : number ,
modLimits : BotModLimits ,
2023-11-13 11:05:05 -05:00
botEquipmentRole : string ,
) : Item [ ]
2023-03-03 15:23:46 +00:00
{
const pmcProfile = this . profileHelper . getPmcProfile ( sessionId ) ;
// Get pool of mods that fit weapon
const compatibleModsPool = modPool [ parentTemplate . _id ] ;
2023-11-13 11:05:05 -05:00
if (
2023-11-13 12:29:16 -05:00
! ( ( parentTemplate . _props . Slots . length || parentTemplate . _props . Cartridges ? . length )
|| parentTemplate . _props . Chambers ? . length )
2023-11-13 11:05:05 -05:00
)
2023-03-03 15:23:46 +00:00
{
2023-11-13 11:05:05 -05:00
this . logger . error (
this . localisationService . getText ( "bot-unable_to_add_mods_to_weapon_missing_ammo_slot" , {
weaponName : parentTemplate._name ,
weaponId : parentTemplate._id ,
2024-02-02 13:54:07 -05:00
botRole : botRole ,
2023-11-13 11:05:05 -05:00
} ) ,
) ;
2023-03-03 15:23:46 +00:00
return weapon ;
}
const botEquipConfig = this . botConfig . equipment [ botEquipmentRole ] ;
2023-11-13 11:05:05 -05:00
const botEquipBlacklist = this . botEquipmentFilterService . getBotEquipmentBlacklist (
botEquipmentRole ,
pmcProfile . Info . Level ,
) ;
2023-03-03 15:23:46 +00:00
const botWeaponSightWhitelist = this . botEquipmentFilterService . getBotWeaponSightWhitelist ( botEquipmentRole ) ;
const randomisationSettings = this . botHelper . getBotRandomizationDetails ( botLevel , botEquipConfig ) ;
2023-10-29 20:45:35 +00:00
// Iterate over mod pool and choose mods to attach
2023-10-10 11:03:20 +00:00
const sortedModKeys = this . sortModKeys ( Object . keys ( compatibleModsPool ) ) ;
2023-03-03 15:23:46 +00:00
for ( const modSlot of sortedModKeys )
{
// Check weapon has slot for mod to fit in
2024-01-25 10:55:33 +00:00
const modsParentSlot = this . getModItemSlotFromDb ( modSlot , parentTemplate ) ;
2023-03-03 15:23:46 +00:00
if ( ! modsParentSlot )
{
2023-11-13 11:05:05 -05:00
this . logger . error (
this . localisationService . getText ( "bot-weapon_missing_mod_slot" , {
modSlot : modSlot ,
weaponId : parentTemplate._id ,
weaponName : parentTemplate._name ,
botRole : botRole ,
} ) ,
) ;
2023-03-03 15:23:46 +00:00
continue ;
}
// Check spawn chance of mod
2024-02-05 22:02:03 +00:00
const modSpawnResult = this . shouldModBeSpawned ( modsParentSlot , modSlot , modSpawnChances , botEquipConfig ) ;
2024-01-25 13:45:42 +00:00
if ( modSpawnResult === ModSpawn . SKIP )
2023-03-03 15:23:46 +00:00
{
continue ;
}
2023-10-29 20:45:35 +00:00
const isRandomisableSlot = randomisationSettings ? . randomisedWeaponModSlots ? . includes ( modSlot ) ? ? false ;
2023-11-13 11:05:05 -05:00
const modToAdd = this . chooseModToPutIntoSlot (
modSlot ,
isRandomisableSlot ,
botWeaponSightWhitelist ,
botEquipBlacklist ,
compatibleModsPool ,
weapon ,
ammoTpl ,
parentTemplate ,
2024-02-02 13:54:07 -05:00
modSpawnResult ,
2023-11-13 11:05:05 -05:00
) ;
2023-03-03 15:23:46 +00:00
// Compatible mod not found
2023-11-13 11:05:05 -05:00
if ( ! modToAdd || typeof modToAdd === "undefined" )
2023-03-03 15:23:46 +00:00
{
continue ;
}
2023-12-28 00:06:45 +00:00
if ( ! this . isModValidForSlot ( modToAdd , modsParentSlot , modSlot , parentTemplate , botRole ) )
2023-03-03 15:23:46 +00:00
{
continue ;
}
2023-10-29 20:45:35 +00:00
const modToAddTemplate = modToAdd [ 1 ] ;
2023-03-03 15:23:46 +00:00
// Skip adding mod to weapon if type limit reached
2023-11-13 11:05:05 -05:00
if (
this . botWeaponModLimitService . weaponModHasReachedLimit (
botEquipmentRole ,
modToAddTemplate ,
modLimits ,
parentTemplate ,
weapon ,
)
)
2023-03-03 15:23:46 +00:00
{
continue ;
}
// If item is a mount for scopes, set scope chance to 100%, this helps fix empty mounts appearing on weapons
if ( this . modSlotCanHoldScope ( modSlot , modToAddTemplate . _parent ) )
{
// mod_mount was picked to be added to weapon, force scope chance to ensure its filled
2023-11-13 12:31:52 -05:00
const scopeSlots = [ "mod_scope" , "mod_scope_000" , "mod_scope_001" , "mod_scope_002" , "mod_scope_003" ] ;
2023-10-10 11:03:20 +00:00
this . adjustSlotSpawnChances ( modSpawnChances , scopeSlots , 100 ) ;
2023-03-03 15:23:46 +00:00
// Hydrate pool of mods that fit into mount as its a randomisable slot
if ( isRandomisableSlot )
{
// Add scope mods to modPool dictionary to ensure the mount has a scope in the pool to pick
this . addCompatibleModsForProvidedMod ( "mod_scope" , modToAddTemplate , modPool , botEquipBlacklist ) ;
}
}
2023-10-10 11:03:20 +00:00
// If picked item is muzzle adapter that can hold a child, adjust spawn chance
if ( this . modSlotCanHoldMuzzleDevices ( modSlot , modToAddTemplate . _parent ) )
{
2023-11-13 12:31:52 -05:00
const muzzleSlots = [ "mod_muzzle" , "mod_muzzle_000" , "mod_muzzle_001" ] ;
2023-10-10 11:03:20 +00:00
// Make chance of muzzle devices 95%, nearly certain but not guaranteed
this . adjustSlotSpawnChances ( modSpawnChances , muzzleSlots , 95 ) ;
}
2023-03-03 15:23:46 +00:00
// If front/rear sight are to be added, set opposite to 100% chance
2023-10-17 20:45:40 +01:00
if ( this . modIsFrontOrRearSight ( modSlot , modToAddTemplate . _id ) )
2023-03-03 15:23:46 +00:00
{
modSpawnChances . mod_sight_front = 100 ;
modSpawnChances . mod_sight_rear = 100 ;
}
2023-10-26 20:31:05 +01:00
// Handguard mod can take a sub handguard mod + weapon has no UBGL (takes same slot)
// Force spawn chance to be 100% to ensure it gets added
2023-11-13 11:05:05 -05:00
if (
2023-11-13 12:29:16 -05:00
modSlot === "mod_handguard" && modToAddTemplate . _props . Slots . find ( ( x ) = > x . _name === "mod_handguard" )
&& ! weapon . find ( ( x ) = > x . slotId === "mod_launcher" )
2023-11-13 11:05:05 -05:00
)
2023-10-26 20:31:05 +01:00
{
// Needed for handguards with lower
modSpawnChances . mod_handguard = 100 ;
}
2023-11-03 17:40:00 +00:00
// If stock mod can take a sub stock mod, force spawn chance to be 100% to ensure sub-stock gets added
// Or if mod_stock is configured to be forced on
2023-11-13 11:05:05 -05:00
if (
2023-11-13 12:31:52 -05:00
modSlot === "mod_stock" && ( modToAddTemplate . _props . Slots . find ( ( x ) = >
2023-11-13 12:29:16 -05:00
x . _name . includes ( "mod_stock" ) || botEquipConfig . forceStock
) )
2023-11-13 11:05:05 -05:00
)
2023-03-03 15:23:46 +00:00
{
// Stock mod can take additional stocks, could be a locking device, force 100% chance
2023-11-03 17:40:00 +00:00
const stockSlots = [ "mod_stock" , "mod_stock_000" , "mod_stock_akms" ] ;
this . adjustSlotSpawnChances ( modSpawnChances , stockSlots , 100 ) ;
2023-03-03 15:23:46 +00:00
}
const modId = this . hashUtil . generate ( ) ;
2024-02-02 13:54:07 -05:00
weapon . push ( this . createModItem ( modId , modToAddTemplate . _id , weaponId , modSlot , modToAddTemplate , botRole ) ) ;
2023-11-13 11:05:05 -05:00
2023-03-03 15:23:46 +00:00
// I first thought we could use the recursive generateModsForItems as previously for cylinder magazines.
2023-11-13 11:05:05 -05:00
// However, the recursion doesn't go over the slots of the parent mod but over the modPool which is given by the bot config
2023-03-03 15:23:46 +00:00
// where we decided to keep cartridges instead of camoras. And since a CylinderMagazine only has one cartridge entry and
// this entry is not to be filled, we need a special handling for the CylinderMagazine
const modParentItem = this . databaseServer . getTables ( ) . templates . items [ modToAddTemplate . _parent ] ;
if ( this . botWeaponGeneratorHelper . magazineIsCylinderRelated ( modParentItem . _name ) )
{
// We don't have child mods, we need to create the camoras for the magazines instead
this . fillCamora ( weapon , modPool , modId , modToAddTemplate ) ;
}
else
{
let containsModInPool = Object . keys ( modPool ) . includes ( modToAddTemplate . _id ) ;
// Sometimes randomised slots are missing sub-mods, if so, get values from mod pool service
// Check for a randomisable slot + without data in modPool + item being added as additional slots
if ( isRandomisableSlot && ! containsModInPool && modToAddTemplate . _props . Slots . length > 0 )
{
const modFromService = this . botEquipmentModPoolService . getModsForWeaponSlot ( modToAddTemplate . _id ) ;
if ( Object . keys ( modFromService ? ? { } ) . length > 0 )
{
modPool [ modToAddTemplate . _id ] = modFromService ;
containsModInPool = true ;
}
}
if ( containsModInPool )
{
2023-11-13 11:05:05 -05:00
// Call self recursively to add mods to this mod
this . generateModsForWeapon (
sessionId ,
weapon ,
modPool ,
modId ,
modToAddTemplate ,
modSpawnChances ,
ammoTpl ,
botRole ,
botLevel ,
modLimits ,
botEquipmentRole ,
) ;
2023-03-03 15:23:46 +00:00
}
}
}
return weapon ;
}
/ * *
* Is this modslot a front or rear sight
* @param modSlot Slot to check
* @returns true if it ' s a front / rear sight
* /
2023-10-17 20:45:40 +01:00
protected modIsFrontOrRearSight ( modSlot : string , tpl : string ) : boolean
2023-03-03 15:23:46 +00:00
{
2023-11-13 11:05:05 -05:00
if ( modSlot === "mod_gas_block" && tpl === "5ae30e795acfc408fb139a0b" )
{ // M4A1 front sight with gas block
2023-10-17 20:45:40 +01:00
return true ;
}
2023-03-03 15:23:46 +00:00
return [ "mod_sight_front" , "mod_sight_rear" ] . includes ( modSlot ) ;
}
/ * *
* Does the provided mod details show the mod can hold a scope
* @param modSlot e . g . mod_scope , mod_mount
* @param modsParentId Parent id of mod item
* @returns true if it can hold a scope
* /
protected modSlotCanHoldScope ( modSlot : string , modsParentId : string ) : boolean
{
2023-11-13 11:05:05 -05:00
return [
"mod_scope" ,
"mod_mount" ,
"mod_mount_000" ,
"mod_scope_000" ,
"mod_scope_001" ,
"mod_scope_002" ,
"mod_scope_003" ,
2023-11-13 12:31:52 -05:00
] . includes ( modSlot . toLowerCase ( ) ) && modsParentId === BaseClasses . MOUNT ;
2023-03-03 15:23:46 +00:00
}
/ * *
2023-10-10 11:03:20 +00:00
* Set mod spawn chances to defined amount
* @param modSpawnChances Chance dictionary to update
2023-03-03 15:23:46 +00:00
* /
2023-11-13 11:05:05 -05:00
protected adjustSlotSpawnChances (
modSpawnChances : ModsChances ,
modSlotsToAdjust : string [ ] ,
newChancePercent : number ,
) : void
2023-03-03 15:23:46 +00:00
{
2023-03-21 14:19:49 +00:00
if ( ! modSpawnChances )
{
this . logger . warning ( "Unable to adjust scope spawn chances as spawn chance object is empty" ) ;
return ;
}
2023-10-10 11:03:20 +00:00
if ( ! modSlotsToAdjust )
{
return ;
}
2023-03-21 14:19:49 +00:00
2023-10-10 11:03:20 +00:00
for ( const modName of modSlotsToAdjust )
2023-03-21 14:19:49 +00:00
{
2023-10-10 11:03:20 +00:00
modSpawnChances [ modName ] = newChancePercent ;
2023-03-21 14:19:49 +00:00
}
2023-03-03 15:23:46 +00:00
}
2023-10-10 11:03:20 +00:00
protected modSlotCanHoldMuzzleDevices ( modSlot : string , modsParentId : string ) : boolean
{
return [ "mod_muzzle" , "mod_muzzle_000" , "mod_muzzle_001" ] . includes ( modSlot . toLowerCase ( ) ) ;
}
2023-03-03 15:23:46 +00:00
protected sortModKeys ( unsortedKeys : string [ ] ) : string [ ]
{
if ( unsortedKeys . length <= 1 )
{
return unsortedKeys ;
}
const sortedKeys : string [ ] = [ ] ;
const modRecieverKey = "mod_reciever" ;
const modMount001Key = "mod_mount_001" ;
2023-10-29 16:35:03 +00:00
const modGasBlockKey = "mod_gas_block" ;
2023-03-03 15:23:46 +00:00
const modPistolGrip = "mod_pistol_grip" ;
const modStockKey = "mod_stock" ;
const modBarrelKey = "mod_barrel" ;
2023-10-29 16:35:03 +00:00
const modHandguardKey = "mod_handguard" ;
2023-03-03 15:23:46 +00:00
const modMountKey = "mod_mount" ;
const modScopeKey = "mod_scope" ;
2023-10-29 16:35:03 +00:00
if ( unsortedKeys . includes ( modHandguardKey ) )
{
sortedKeys . push ( modHandguardKey ) ;
unsortedKeys . splice ( unsortedKeys . indexOf ( modHandguardKey ) , 1 ) ;
}
2023-03-03 15:23:46 +00:00
if ( unsortedKeys . includes ( modBarrelKey ) )
{
sortedKeys . push ( modBarrelKey ) ;
unsortedKeys . splice ( unsortedKeys . indexOf ( modBarrelKey ) , 1 ) ;
}
if ( unsortedKeys . includes ( modMount001Key ) )
{
sortedKeys . push ( modMount001Key ) ;
unsortedKeys . splice ( unsortedKeys . indexOf ( modMount001Key ) , 1 ) ;
}
if ( unsortedKeys . includes ( modRecieverKey ) )
{
sortedKeys . push ( modRecieverKey ) ;
unsortedKeys . splice ( unsortedKeys . indexOf ( modRecieverKey ) , 1 ) ;
}
2023-11-13 11:05:05 -05:00
2023-03-03 15:23:46 +00:00
if ( unsortedKeys . includes ( modPistolGrip ) )
{
sortedKeys . push ( modPistolGrip ) ;
unsortedKeys . splice ( unsortedKeys . indexOf ( modPistolGrip ) , 1 ) ;
}
2023-10-29 16:35:03 +00:00
if ( unsortedKeys . includes ( modGasBlockKey ) )
2023-03-03 15:23:46 +00:00
{
2023-10-29 16:35:03 +00:00
sortedKeys . push ( modGasBlockKey ) ;
unsortedKeys . splice ( unsortedKeys . indexOf ( modGasBlockKey ) , 1 ) ;
2023-03-03 15:23:46 +00:00
}
if ( unsortedKeys . includes ( modStockKey ) )
{
sortedKeys . push ( modStockKey ) ;
unsortedKeys . splice ( unsortedKeys . indexOf ( modStockKey ) , 1 ) ;
}
if ( unsortedKeys . includes ( modMountKey ) )
{
sortedKeys . push ( modMountKey ) ;
unsortedKeys . splice ( unsortedKeys . indexOf ( modMountKey ) , 1 ) ;
}
if ( unsortedKeys . includes ( modScopeKey ) )
{
sortedKeys . push ( modScopeKey ) ;
unsortedKeys . splice ( unsortedKeys . indexOf ( modScopeKey ) , 1 ) ;
}
sortedKeys . push ( . . . unsortedKeys ) ;
return sortedKeys ;
}
/ * *
* Get a Slot property for an item ( chamber / cartridge / slot )
* @param modSlot e . g patron_in_weapon
* @param parentTemplate item template
* @returns Slot item
* /
2024-01-25 10:55:33 +00:00
protected getModItemSlotFromDb ( modSlot : string , parentTemplate : ITemplateItem ) : Slot
2023-03-03 15:23:46 +00:00
{
2023-12-27 23:29:37 +00:00
const modSlotLower = modSlot . toLowerCase ( ) ;
switch ( modSlotLower )
2023-03-03 15:23:46 +00:00
{
case "patron_in_weapon" :
case "patron_in_weapon_000" :
case "patron_in_weapon_001" :
2023-12-27 23:29:37 +00:00
return parentTemplate . _props . Chambers . find ( ( chamber ) = > chamber . _name . includes ( modSlotLower ) ) ;
2023-03-03 15:23:46 +00:00
case "cartridges" :
2023-12-27 23:29:37 +00:00
return parentTemplate . _props . Cartridges . find ( ( c ) = > c . _name . toLowerCase ( ) === modSlotLower ) ;
2023-03-03 15:23:46 +00:00
default :
2023-12-27 23:29:37 +00:00
return parentTemplate . _props . Slots . find ( ( s ) = > s . _name . toLowerCase ( ) === modSlotLower ) ;
2023-03-03 15:23:46 +00:00
}
}
/ * *
2023-03-21 14:19:49 +00:00
* Randomly choose if a mod should be spawned , 100 % for required mods OR mod is ammo slot
2023-03-03 15:23:46 +00:00
* @param itemSlot slot the item sits in
* @param modSlot slot the mod sits in
* @param modSpawnChances Chances for various mod spawns
2024-02-05 22:02:03 +00:00
* @param botEquipConfig Various config settings for generating this type of bot
2024-01-25 13:45:42 +00:00
* @returns ModSpawn . SPAWN when mod should be spawned , ModSpawn . DEFAULT_MOD when default mod should spawn , ModSpawn . SKIP when mod is skipped
2023-03-03 15:23:46 +00:00
* /
2024-02-05 22:02:03 +00:00
protected shouldModBeSpawned (
itemSlot : Slot ,
modSlot : string ,
modSpawnChances : ModsChances ,
botEquipConfig : EquipmentFilters ,
) : ModSpawn
2023-03-03 15:23:46 +00:00
{
2024-01-25 13:45:42 +00:00
const slotRequired = itemSlot . _required ;
if ( this . getAmmoContainers ( ) . includes ( modSlot ) )
2023-03-03 15:23:46 +00:00
{
2024-02-02 13:54:07 -05:00
return ModSpawn . SPAWN ;
2024-01-25 13:45:42 +00:00
}
const spawnMod = this . probabilityHelper . rollChance ( modSpawnChances [ modSlot ] ) ;
2024-02-05 22:02:03 +00:00
if ( ! spawnMod && ( slotRequired || botEquipConfig . weaponSlotIdsToMakeRequired . includes ( modSlot ) ) )
2024-01-25 13:45:42 +00:00
{
// Mod is required but spawn chance roll failed, choose default mod spawn for slot
return ModSpawn . DEFAULT_MOD ;
2023-03-03 15:23:46 +00:00
}
2024-02-02 13:54:07 -05:00
return spawnMod ? ModSpawn.SPAWN : ModSpawn.SKIP ;
2023-03-03 15:23:46 +00:00
}
/ * *
* @param modSlot Slot mod will fit into
* @param isRandomisableSlot Will generate a randomised mod pool if true
* @param modsParent Parent slot the item will be a part of
* @param botEquipBlacklist Blacklist to prevent mods from being picked
* @param itemModPool Pool of items to pick from
* @param weapon array with only weapon tpl in it , ready for mods to be added
* @param ammoTpl ammo tpl to use if slot requires a cartridge to be added ( e . g . mod_magazine )
* @param parentTemplate Parent item the mod will go into
2024-01-25 13:45:42 +00:00
* @returns itemHelper . getItem ( ) result
2023-03-03 15:23:46 +00:00
* /
protected chooseModToPutIntoSlot (
modSlot : string ,
isRandomisableSlot : boolean ,
botWeaponSightWhitelist : Record < string , string [ ] > ,
botEquipBlacklist : EquipmentFilterDetails ,
itemModPool : Record < string , string [ ] > ,
weapon : Item [ ] ,
ammoTpl : string ,
2023-11-13 11:05:05 -05:00
parentTemplate : ITemplateItem ,
2024-02-02 13:54:07 -05:00
modSpawnResult : ModSpawn ,
2023-11-13 11:05:05 -05:00
) : [ boolean , ITemplateItem ]
2023-03-03 15:23:46 +00:00
{
2024-01-25 13:45:42 +00:00
/** Slot mod will fill */
2023-11-13 11:05:05 -05:00
const parentSlot = parentTemplate . _props . Slots . find ( ( i ) = > i . _name === modSlot ) ;
2024-01-25 13:45:42 +00:00
const weaponTemplate = this . itemHelper . getItem ( weapon [ 0 ] . _tpl ) [ 1 ] ;
2023-11-13 11:05:05 -05:00
2023-03-03 15:23:46 +00:00
// It's ammo, use predefined ammo parameter
if ( this . getAmmoContainers ( ) . includes ( modSlot ) && modSlot !== "mod_magazine" )
{
2024-02-02 13:54:07 -05:00
return this . itemHelper . getItem ( ammoTpl ) ;
2023-03-03 15:23:46 +00:00
}
2024-02-02 13:54:07 -05:00
2024-01-27 18:12:13 +00:00
// Ensure there's a pool of mods to pick from
2024-02-02 13:54:07 -05:00
let modPool = this . getModPoolForSlot (
itemModPool ,
modSpawnResult ,
parentTemplate ,
weaponTemplate ,
modSlot ,
botEquipBlacklist ,
isRandomisableSlot ,
) ;
2024-01-27 18:12:13 +00:00
if ( ! ( modPool || parentSlot . _required ) )
2023-03-03 15:23:46 +00:00
{
2024-01-27 18:12:13 +00:00
// Nothing in mod pool + item not required
2024-02-02 13:54:07 -05:00
this . logger . debug ( ` Mod pool for slot: ${ modSlot } on item: ${ parentTemplate . _name } was empty, skipping mod ` ) ;
2024-01-27 18:12:13 +00:00
return null ;
}
2023-03-03 15:23:46 +00:00
2024-01-27 18:12:13 +00:00
// Filter out non-whitelisted scopes, use full modpool if filtered pool would have no elements
if ( modSlot . includes ( "mod_scope" ) && botWeaponSightWhitelist )
{
// scope pool has more than one scope
if ( modPool . length > 1 )
2023-03-03 15:23:46 +00:00
{
2024-02-02 13:54:07 -05:00
modPool = this . filterSightsByWeaponType ( weapon [ 0 ] , modPool , botWeaponSightWhitelist ) ;
2023-03-03 15:23:46 +00:00
}
2024-01-27 18:12:13 +00:00
}
2024-02-02 13:54:07 -05:00
2024-01-27 18:12:13 +00:00
// Pick random mod that's compatible
2024-02-02 13:54:07 -05:00
const chosenModResult = this . pickWeaponModTplForSlotFromPool (
modPool ,
parentSlot ,
modSpawnResult ,
weapon ,
modSlot ,
) ;
2024-01-29 22:41:08 +00:00
if ( chosenModResult . slotBlocked && ! parentSlot . _required )
2024-01-27 18:12:13 +00:00
{
// Don't bother trying to fit mod, slot is completely blocked
return null ;
}
// Log if mod chosen was incompatible
if ( chosenModResult . incompatible && parentSlot . _required )
{
this . logger . debug ( chosenModResult . reason ) ;
2024-02-02 13:54:07 -05:00
// this.logger.debug(`Weapon: ${weapon.map(x => `${x._tpl} ${x.slotId ?? ""}`).join(",")}`)
2023-03-03 15:23:46 +00:00
}
// Get random mod to attach from items db for required slots if none found above
2024-01-27 18:12:13 +00:00
if ( ! chosenModResult . found && parentSlot !== undefined && parentSlot . _required )
2023-03-03 15:23:46 +00:00
{
2024-01-27 18:12:13 +00:00
chosenModResult . chosenTpl = this . getRandomModTplFromItemDb ( "" , parentSlot , modSlot , weapon ) ;
chosenModResult . found = true ;
2023-03-03 15:23:46 +00:00
}
// Compatible item not found + not required
2024-01-27 18:12:13 +00:00
if ( ! chosenModResult . found && parentSlot !== undefined && ! parentSlot . _required )
2023-03-03 15:23:46 +00:00
{
return null ;
}
2024-01-27 18:12:13 +00:00
if ( ! chosenModResult . found && parentSlot !== undefined )
2023-03-03 15:23:46 +00:00
{
if ( parentSlot . _required )
{
2023-11-13 11:05:05 -05:00
this . logger . warning (
` Required slot unable to be filled, ${ modSlot } on ${ parentTemplate . _name } ${ parentTemplate . _id } for weapon: ${
weapon [ 0 ] . _tpl
} ` ,
) ;
2023-03-03 15:23:46 +00:00
}
return null ;
}
2024-01-27 18:12:13 +00:00
return this . itemHelper . getItem ( chosenModResult . chosenTpl ) ;
}
2024-01-29 22:41:08 +00:00
protected pickWeaponModTplForSlotFromPool (
2024-01-27 18:12:13 +00:00
modPool : string [ ] ,
parentSlot : Slot ,
modSpawnResult : ModSpawn ,
weapon : Item [ ] ,
2024-02-02 13:54:07 -05:00
modSlotname : string ,
) : IChooseRandomCompatibleModResult
2024-01-27 18:12:13 +00:00
{
let chosenTpl : string ;
const exhaustableModPool = new ExhaustableArray ( modPool , this . randomUtil , this . jsonUtil ) ;
2024-02-02 13:54:07 -05:00
let chosenModResult : IChooseRandomCompatibleModResult = { incompatible : true , found : false , reason : "unknown" } ;
2024-01-27 18:12:13 +00:00
const modParentFilterList = parentSlot . _props . filters [ 0 ] . Filter ;
2024-01-29 23:02:09 +00:00
// How many times can a mod for the slot be blocked before we stop trying
const maxBlockedAttempts = Math . round ( modPool . length * 0.75 ) ; // Roughly 75% of pool size
2024-01-29 22:41:08 +00:00
let blockedAttemptCount = 0 ;
2024-01-27 18:12:13 +00:00
while ( exhaustableModPool . hasValues ( ) )
{
chosenTpl = exhaustableModPool . getRandomValue ( ) ;
if ( modSpawnResult === ModSpawn . DEFAULT_MOD && modPool . length === 1 )
{
// Default mod wanted and only one choice in pool
chosenModResult . found = true ;
2024-01-29 21:27:36 +00:00
chosenModResult . incompatible = false ;
2024-01-27 18:12:13 +00:00
chosenModResult . chosenTpl = chosenTpl ;
break ;
}
2024-01-29 23:02:09 +00:00
// Check chosen item is on the allowed list of the parent
2024-01-27 18:12:13 +00:00
const isOnModParentFilterList = modParentFilterList . includes ( chosenTpl ) ;
if ( ! isOnModParentFilterList )
{
2024-01-29 23:02:09 +00:00
// Try again
2024-01-27 18:12:13 +00:00
continue ;
}
2024-01-29 22:41:08 +00:00
chosenModResult = this . botGeneratorHelper . isWeaponModIncompatibleWithCurrentMods (
2024-01-27 18:12:13 +00:00
weapon ,
chosenTpl ,
modSlotname ,
) ;
if ( chosenModResult . slotBlocked )
{
2024-01-30 21:14:15 +00:00
// Give max of x attempts of picking a mod if blocked by another
2024-01-29 23:02:09 +00:00
if ( blockedAttemptCount > maxBlockedAttempts )
2024-01-29 22:41:08 +00:00
{
blockedAttemptCount = 0 ;
break ;
}
blockedAttemptCount ++ ;
2024-01-29 23:02:09 +00:00
// Try again
2024-01-29 22:41:08 +00:00
continue ;
2024-01-27 18:12:13 +00:00
}
2024-02-02 13:54:07 -05:00
2024-01-29 23:02:09 +00:00
// Some mod combos will never work, make sure this isnt the case
2024-02-02 15:00:12 -05:00
if ( ! ( chosenModResult . incompatible || this . weaponModComboIsIncompatible ( weapon , chosenTpl ) ) )
2024-01-27 18:12:13 +00:00
{
2024-01-29 23:02:09 +00:00
// Success
2024-01-27 18:12:13 +00:00
chosenModResult . found = true ;
2024-01-29 22:41:08 +00:00
chosenModResult . incompatible = false ;
2024-01-27 18:12:13 +00:00
chosenModResult . chosenTpl = chosenTpl ;
break ;
}
}
return chosenModResult ;
2023-03-03 15:23:46 +00:00
}
2024-01-25 13:45:42 +00:00
/ * *
* Filter mod pool down based on various criteria :
* Is slot flagged as randomisable
* Is slot required
* Is slot flagged as default mod only
* @param itemModPool Existing pool of mods to choose
* @param modSpawnResult outcome of random roll to select if mod should be added
* @param parentTemplate Mods parent
* @param weaponTemplate Mods root parent ( weapon / equipment )
* @param modSlot name of mod slot to choose for
2024-02-02 13:54:07 -05:00
* @param botEquipBlacklist
2024-01-25 13:45:42 +00:00
* @param isRandomisableSlot is flagged as a randomisable slot
2024-02-02 13:54:07 -05:00
* @returns
2024-01-25 13:45:42 +00:00
* /
protected getModPoolForSlot (
itemModPool : Record < string , string [ ] > ,
modSpawnResult : ModSpawn ,
parentTemplate : ITemplateItem ,
weaponTemplate : ITemplateItem ,
modSlot : string ,
botEquipBlacklist : EquipmentFilterDetails ,
2024-02-02 13:54:07 -05:00
isRandomisableSlot : boolean ,
) : string [ ]
2024-01-25 13:45:42 +00:00
{
// Mod is flagged as being default only, try and find it in globals
if ( modSpawnResult === ModSpawn . DEFAULT_MOD )
{
2024-02-05 15:36:05 +00:00
const matchingPreset = this . getMatchingPreset ( weaponTemplate , parentTemplate . _id ) ;
const matchingMod = matchingPreset . _items . find ( ( item ) = >
2024-02-02 13:54:07 -05:00
item ? . slotId ? . toLowerCase ( ) === modSlot . toLowerCase ( )
) ;
2024-01-30 21:14:15 +00:00
// Only filter mods down to single default item if it already exists in existing itemModPool, OR the default item has no children
// Filtering mod pool to item that wasnt already there can have problems;
// You'd have a mod being picked without any sub-mods in its chain, possibly resulting in missing required mods not being added
2024-01-25 13:45:42 +00:00
if ( matchingMod )
{
2024-01-30 21:14:15 +00:00
// Mod isnt in existing mod pool
if ( itemModPool [ modSlot ] . includes ( matchingMod . _tpl ) )
{
// Found mod on preset + it already exists in mod pool
return [ matchingMod . _tpl ] ;
}
// Mod isnt in existing pool, only add if its got no children
if ( this . itemHelper . getItem ( matchingMod . _tpl ) [ 1 ] . _props . Slots . length === 0 )
{
// Mod has no children
return [ matchingMod . _tpl ] ;
}
2024-01-25 13:45:42 +00:00
}
2024-01-30 21:14:15 +00:00
this . logger . debug ( ` No default: ${ modSlot } mod found on template: ${ weaponTemplate . _id } ` ) ;
2024-01-25 13:45:42 +00:00
2024-01-30 21:14:15 +00:00
// Couldnt find default in globals, use existing mod pool data
2024-01-25 13:45:42 +00:00
return itemModPool [ modSlot ] ;
}
if ( isRandomisableSlot )
{
return this . getDynamicModPool ( parentTemplate . _id , modSlot , botEquipBlacklist ) ;
}
// Required mod is not default or randomisable, use existing pool
return itemModPool [ modSlot ] ;
}
2024-02-05 15:36:05 +00:00
/ * *
* Get default preset for weapon , get specific weapon presets for edge cases ( mp5 / silenced dvl )
* @param weaponTemplate
* @param parentItemTpl
* @returns
* /
protected getMatchingPreset ( weaponTemplate : ITemplateItem , parentItemTpl : string ) : IPreset
{
// Edge case - using mp5sd reciever means default mp5 handguard doesnt fit
const isMp5sd = parentItemTpl === "5926f2e086f7745aae644231" ;
// Edge case - dvl 500mm is the silenced barrel and has specific muzzle mods
const isDvl500mmSilencedBarrel = parentItemTpl === "5888945a2459774bf43ba385" ;
if ( isMp5sd )
{
return this . presetHelper . getPreset ( "59411abb86f77478f702b5d2" ) ;
}
if ( isDvl500mmSilencedBarrel )
{
return this . presetHelper . getPreset ( "59e8d2b386f77445830dd299" ) ;
}
return this . presetHelper . getDefaultPreset ( weaponTemplate . _id ) ;
}
2023-11-17 12:40:23 +00:00
/ * *
* Temp fix to prevent certain combinations of weapons with mods that are known to be incompatible
* @param weapon Weapon
* @param modTpl Mod to check compatibility with weapon
* @returns True if incompatible
* /
protected weaponModComboIsIncompatible ( weapon : Item [ ] , modTpl : string ) : boolean
{
// STM-9 + AR-15 Lone Star Ion Lite handguard
if ( weapon [ 0 ] . _tpl === "60339954d62c9b14ed777c06" && modTpl === "5d4405f0a4b9361e6a4e6bd9" )
{
return true ;
}
return false ;
}
2023-03-03 15:23:46 +00:00
/ * *
* Create a mod item with parameters as properties
* @param modId _id
* @param modTpl _tpl
* @param parentId parentId
* @param modSlot slotId
2023-11-13 11:05:05 -05:00
* @param modTemplate Used to add additional properties in the upd object
2023-03-03 15:23:46 +00:00
* @returns Item object
* /
2023-11-13 11:05:05 -05:00
protected createModItem (
modId : string ,
modTpl : string ,
parentId : string ,
modSlot : string ,
modTemplate : ITemplateItem ,
botRole : string ,
) : Item
2023-03-03 15:23:46 +00:00
{
return {
2023-10-31 22:52:09 +00:00
_id : modId ,
_tpl : modTpl ,
parentId : parentId ,
slotId : modSlot ,
2023-11-13 11:05:05 -05:00
. . . this . botGeneratorHelper . generateExtraPropertiesForItem ( modTemplate , botRole ) ,
2023-03-03 15:23:46 +00:00
} ;
}
/ * *
* Get a list of containers that hold ammo
* e . g . mod_magazine / patron_in_weapon_000
* @returns string array
* /
protected getAmmoContainers ( ) : string [ ]
{
return [ "mod_magazine" , "patron_in_weapon" , "patron_in_weapon_000" , "patron_in_weapon_001" , "cartridges" ] ;
}
/ * *
* Get a random mod from an items compatible mods Filter array
* @param modTpl ? ? ? ? default value to return if nothing found
2023-11-13 11:05:05 -05:00
* @param parentSlot item mod will go into , used to get compatible items
2023-03-03 15:23:46 +00:00
* @param modSlot Slot to get mod to fill
* @param items items to ensure picked mod is compatible with
* @returns item tpl
* /
2024-01-25 13:45:42 +00:00
protected getRandomModTplFromItemDb ( modTpl : string , parentSlot : Slot , modSlot : string , items : Item [ ] ) : string
2023-03-03 15:23:46 +00:00
{
2023-11-13 11:05:05 -05:00
// Find compatible mods and make an array of them
2023-03-03 15:23:46 +00:00
const allowedItems = parentSlot . _props . filters [ 0 ] . Filter ;
// Find mod item that fits slot from sorted mod array
const exhaustableModPool = new ExhaustableArray ( allowedItems , this . randomUtil , this . jsonUtil ) ;
let tmpModTpl = modTpl ;
while ( exhaustableModPool . hasValues ( ) )
{
tmpModTpl = exhaustableModPool . getRandomValue ( ) ;
if ( ! this . botGeneratorHelper . isItemIncompatibleWithCurrentItems ( items , tmpModTpl , modSlot ) . incompatible )
{
return tmpModTpl ;
}
}
// No mod found
return null ;
}
/ * *
* Log errors if mod is not compatible with slot
* @param modToAdd template of mod to check
2024-01-27 18:12:13 +00:00
* @param slotAddedToTemplate slot the item will be placed in
2023-03-03 15:23:46 +00:00
* @param modSlot slot the mod will fill
2024-01-27 18:12:13 +00:00
* @param parentTemplate template of the mods being added
2023-12-28 00:06:45 +00:00
* @param botRole
2023-03-03 15:23:46 +00:00
* @returns true if valid
* /
2023-11-13 11:05:05 -05:00
protected isModValidForSlot (
modToAdd : [ boolean , ITemplateItem ] ,
2024-01-27 18:12:13 +00:00
slotAddedToTemplate : Slot ,
2023-11-13 11:05:05 -05:00
modSlot : string ,
parentTemplate : ITemplateItem ,
2024-02-02 13:54:07 -05:00
botRole : string ,
2023-11-13 11:05:05 -05:00
) : boolean
2023-03-03 15:23:46 +00:00
{
2024-01-27 18:12:13 +00:00
const modBeingAddedTemplate = modToAdd [ 1 ] ;
2023-03-03 15:23:46 +00:00
// Mod lacks template item
2024-01-27 18:12:13 +00:00
if ( ! modBeingAddedTemplate )
2023-03-03 15:23:46 +00:00
{
2023-11-13 11:05:05 -05:00
this . logger . error (
this . localisationService . getText ( "bot-no_item_template_found_when_adding_mod" , {
2024-01-27 18:12:13 +00:00
modId : modBeingAddedTemplate._id ,
2023-11-13 11:05:05 -05:00
modSlot : modSlot ,
} ) ,
) ;
2023-03-03 15:23:46 +00:00
this . logger . debug ( ` Item -> ${ parentTemplate . _id } ; Slot -> ${ modSlot } ` ) ;
return false ;
}
// Mod isn't a valid item
if ( ! modToAdd [ 0 ] )
{
// Slot must be filled, show warning
2024-01-27 18:12:13 +00:00
if ( slotAddedToTemplate . _required )
2023-03-03 15:23:46 +00:00
{
2023-11-13 11:05:05 -05:00
this . logger . warning (
this . localisationService . getText ( "bot-unable_to_add_mod_item_invalid" , {
2024-01-27 18:12:13 +00:00
itemName : modBeingAddedTemplate._name ,
2023-11-13 11:05:05 -05:00
modSlot : modSlot ,
parentItemName : parentTemplate._name ,
2024-02-02 13:54:07 -05:00
botRole : botRole ,
2023-11-13 11:05:05 -05:00
} ) ,
) ;
2023-03-03 15:23:46 +00:00
}
return false ;
}
return true ;
}
/ * *
* Find mod tpls of a provided type and add to modPool
* @param desiredSlotName slot to look up and add we are adding tpls for ( e . g mod_scope )
* @param modTemplate db object for modItem we get compatible mods from
* @param modPool Pool of mods we are adding to
* /
2023-11-13 11:05:05 -05:00
protected addCompatibleModsForProvidedMod (
desiredSlotName : string ,
modTemplate : ITemplateItem ,
modPool : Mods ,
botEquipBlacklist : EquipmentFilterDetails ,
) : void
2023-03-03 15:23:46 +00:00
{
2023-11-13 11:05:05 -05:00
const desiredSlotObject = modTemplate . _props . Slots . find ( ( x ) = > x . _name . includes ( desiredSlotName ) ) ;
2023-03-03 15:23:46 +00:00
if ( desiredSlotObject )
{
const supportedSubMods = desiredSlotObject . _props . filters [ 0 ] . Filter ;
if ( supportedSubMods )
{
// Filter mods
2023-11-13 11:05:05 -05:00
let filteredMods = this . filterWeaponModsByBlacklist (
supportedSubMods ,
botEquipBlacklist ,
desiredSlotName ,
) ;
2023-03-03 15:23:46 +00:00
if ( filteredMods . length === 0 )
{
2023-11-13 11:05:05 -05:00
this . logger . warning (
this . localisationService . getText ( "bot-unable_to_filter_mods_all_blacklisted" , {
slotName : desiredSlotObject._name ,
itemName : modTemplate._name ,
} ) ,
) ;
2023-03-03 15:23:46 +00:00
filteredMods = supportedSubMods ;
}
if ( ! modPool [ modTemplate . _id ] )
{
modPool [ modTemplate . _id ] = { } ;
}
modPool [ modTemplate . _id ] [ desiredSlotObject . _name ] = supportedSubMods ;
}
}
}
/ * *
* Get the possible items that fit a slot
* @param parentItemId item tpl to get compatible items for
* @param modSlot Slot item should fit in
* @param botEquipBlacklist equipment that should not be picked
* @returns array of compatible items for that slot
* /
2023-11-13 11:05:05 -05:00
protected getDynamicModPool (
parentItemId : string ,
modSlot : string ,
botEquipBlacklist : EquipmentFilterDetails ,
) : string [ ]
2023-03-03 15:23:46 +00:00
{
2023-11-13 11:05:05 -05:00
const modsFromDynamicPool = this . jsonUtil . clone (
this . botEquipmentModPoolService . getCompatibleModsForWeaponSlot ( parentItemId , modSlot ) ,
) ;
2023-03-03 15:23:46 +00:00
const filteredMods = this . filterWeaponModsByBlacklist ( modsFromDynamicPool , botEquipBlacklist , modSlot ) ;
if ( filteredMods . length === 0 )
{
2023-11-13 11:05:05 -05:00
this . logger . warning (
this . localisationService . getText ( "bot-unable_to_filter_mod_slot_all_blacklisted" , modSlot ) ,
) ;
2023-03-03 15:23:46 +00:00
return modsFromDynamicPool ;
}
return filteredMods ;
}
/ * *
* Take a list of tpls and filter out blacklisted values using itemFilterService + botEquipmentBlacklist
* @param allowedMods base mods to filter
* @param botEquipBlacklist equipment blacklist
* @param modSlot slot mods belong to
* @returns Filtered array of mod tpls
* /
2023-11-13 11:05:05 -05:00
protected filterWeaponModsByBlacklist (
allowedMods : string [ ] ,
botEquipBlacklist : EquipmentFilterDetails ,
modSlot : string ,
) : string [ ]
2023-03-03 15:23:46 +00:00
{
if ( ! botEquipBlacklist )
{
return allowedMods ;
}
2023-11-13 11:05:05 -05:00
2023-03-03 15:23:46 +00:00
let result : string [ ] = [ ] ;
2023-11-13 11:05:05 -05:00
// Get item blacklist and mod equipment blacklist as one array
const blacklist = this . itemFilterService . getBlacklistedItems ( ) . concat (
botEquipBlacklist . equipment [ modSlot ] || [ ] ,
) ;
result = allowedMods . filter ( ( x ) = > ! blacklist . includes ( x ) ) ;
2023-03-03 15:23:46 +00:00
return result ;
}
/ * *
* With the shotgun revolver ( 60 db29ce99594040e04c4a27 ) 12.12 introduced CylinderMagazines .
* Those magazines ( e . g . 60 dc519adf4c47305f6d410d ) have a "Cartridges" entry with a _max_count = 0 .
* Ammo is not put into the magazine directly but assigned to the magazine ' s slots : The "camora_xxx" slots .
* This function is a helper called by generateModsForItem for mods with parent type "CylinderMagazine"
* @param items The items where the CylinderMagazine ' s camora are appended to
2023-11-04 20:11:09 +00:00
* @param modPool modPool which should include available cartridges
2023-03-03 15:23:46 +00:00
* @param parentId The CylinderMagazine ' s UID
* @param parentTemplate The CylinderMagazine ' s template
* /
protected fillCamora ( items : Item [ ] , modPool : Mods , parentId : string , parentTemplate : ITemplateItem ) : void
{
2023-11-04 20:11:09 +00:00
let itemModPool = modPool [ parentTemplate . _id ] ;
2023-03-03 15:23:46 +00:00
if ( ! itemModPool )
{
2023-11-13 11:05:05 -05:00
this . logger . warning (
this . localisationService . getText ( "bot-unable_to_fill_camora_slot_mod_pool_empty" , {
weaponId : parentTemplate._id ,
weaponName : parentTemplate._name ,
} ) ,
) ;
const camoraSlots = parentTemplate . _props . Slots . filter ( ( x ) = > x . _name . startsWith ( "camora" ) ) ;
2023-11-04 20:11:09 +00:00
// Attempt to generate camora slots for item
modPool [ parentTemplate . _id ] = { } ;
for ( const camora of camoraSlots )
{
modPool [ parentTemplate . _id ] [ camora . _name ] = camora . _props . filters [ 0 ] . Filter ;
}
itemModPool = modPool [ parentTemplate . _id ] ;
2023-03-03 15:23:46 +00:00
}
let exhaustableModPool = null ;
let modSlot = "cartridges" ;
const camoraFirstSlot = "camora_000" ;
if ( modSlot in itemModPool )
{
exhaustableModPool = new ExhaustableArray ( itemModPool [ modSlot ] , this . randomUtil , this . jsonUtil ) ;
}
else if ( camoraFirstSlot in itemModPool )
{
modSlot = camoraFirstSlot ;
2023-11-13 11:05:05 -05:00
exhaustableModPool = new ExhaustableArray (
this . mergeCamoraPoolsTogether ( itemModPool ) ,
this . randomUtil ,
this . jsonUtil ,
) ;
2023-03-03 15:23:46 +00:00
}
else
{
this . logger . error ( this . localisationService . getText ( "bot-missing_cartridge_slot" , parentTemplate . _id ) ) ;
return ;
}
let modTpl : string ;
let found = false ;
while ( exhaustableModPool . hasValues ( ) )
{
modTpl = exhaustableModPool . getRandomValue ( ) ;
if ( ! this . botGeneratorHelper . isItemIncompatibleWithCurrentItems ( items , modTpl , modSlot ) . incompatible )
{
found = true ;
break ;
}
}
if ( ! found )
{
this . logger . error ( this . localisationService . getText ( "bot-no_compatible_camora_ammo_found" , modSlot ) ) ;
return ;
}
for ( const slot of parentTemplate . _props . Slots )
{
const modSlotId = slot . _name ;
const modId = this . hashUtil . generate ( ) ;
2023-11-13 12:38:16 -05:00
items . push ( { _id : modId , _tpl : modTpl , parentId : parentId , slotId : modSlotId } ) ;
2023-03-03 15:23:46 +00:00
}
}
/ * *
2023-11-13 11:05:05 -05:00
* Take a record of camoras and merge the compatible shells into one array
2023-03-03 15:23:46 +00:00
* @param camorasWithShells camoras we want to merge into one array
2023-11-13 11:05:05 -05:00
* @returns string array of shells for multiple camora sources
2023-03-03 15:23:46 +00:00
* /
2023-11-13 11:05:05 -05:00
protected mergeCamoraPoolsTogether ( camorasWithShells : Record < string , string [ ] > ) : string [ ]
2023-03-03 15:23:46 +00:00
{
const poolResult : string [ ] = [ ] ;
for ( const camoraKey in camorasWithShells )
{
const shells = camorasWithShells [ camoraKey ] ;
for ( const shell of shells )
{
// Only add distinct shells
if ( ! poolResult . includes ( shell ) )
{
poolResult . push ( shell ) ;
}
}
}
return poolResult ;
}
/ * *
* Filter out non - whitelisted weapon scopes
2023-10-10 11:03:20 +00:00
* Controlled by bot . json weaponSightWhitelist
* e . g . filter out rifle scopes from SMGs
2023-03-03 15:23:46 +00:00
* @param weapon Weapon scopes will be added to
* @param scopes Full scope pool
2023-11-13 11:05:05 -05:00
* @param botWeaponSightWhitelist Whitelist of scope types by weapon base type
2023-10-10 11:03:20 +00:00
* @returns Array of scope tpls that have been filtered to just ones allowed for that weapon type
2023-03-03 15:23:46 +00:00
* /
2023-11-13 11:05:05 -05:00
protected filterSightsByWeaponType (
weapon : Item ,
scopes : string [ ] ,
botWeaponSightWhitelist : Record < string , string [ ] > ,
) : string [ ]
2023-03-03 15:23:46 +00:00
{
const weaponDetails = this . itemHelper . getItem ( weapon . _tpl ) ;
// Return original scopes array if whitelist not found
const whitelistedSightTypes = botWeaponSightWhitelist [ weaponDetails [ 1 ] . _parent ] ;
if ( ! whitelistedSightTypes )
{
2023-11-13 11:05:05 -05:00
this . logger . debug (
` Unable to find whitelist for weapon type: ${ weaponDetails [ 1 ] . _parent } ${
weaponDetails [ 1 ] . _name
} , skipping sight filtering ` ,
) ;
2023-10-10 11:03:20 +00:00
2023-03-03 15:23:46 +00:00
return scopes ;
}
// Filter items that are not directly scopes OR mounts that do not hold the type of scope we allow for this weapon type
const filteredScopesAndMods : string [ ] = [ ] ;
for ( const item of scopes )
{
// Mods is a scope, check base class is allowed
if ( this . itemHelper . isOfBaseclasses ( item , whitelistedSightTypes ) )
{
2023-10-10 11:03:20 +00:00
// Add mod to allowed list
2023-03-03 15:23:46 +00:00
filteredScopesAndMods . push ( item ) ;
continue ;
}
2023-10-10 11:03:20 +00:00
// Edge case, what if item is a mount for a scope and not directly a scope?
// Check item is mount + has child items
2023-03-03 15:23:46 +00:00
const itemDetails = this . itemHelper . getItem ( item ) [ 1 ] ;
2023-11-13 11:05:05 -05:00
if ( this . itemHelper . isOfBaseclass ( item , BaseClasses . MOUNT ) && itemDetails . _props . Slots . length > 0 )
2023-03-03 15:23:46 +00:00
{
2023-10-10 11:03:20 +00:00
// Check to see if mount has a scope slot (only include primary slot, ignore the rest like the backup sight slots)
// Should only find 1 as there's currently no items with a mod_scope AND a mod_scope_000
2023-11-13 11:05:05 -05:00
const scopeSlot = itemDetails . _props . Slots . filter ( ( x ) = >
[ "mod_scope" , "mod_scope_000" ] . includes ( x . _name )
) ;
2023-10-10 11:03:20 +00:00
// Mods scope slot found must allow ALL whitelisted scope types OR be a mount
2023-11-13 11:05:05 -05:00
if (
scopeSlot ? . every ( ( x ) = >
x . _props . filters [ 0 ] . Filter . every ( ( x ) = >
2023-11-13 12:29:16 -05:00
this . itemHelper . isOfBaseclasses ( x , whitelistedSightTypes )
|| this . itemHelper . isOfBaseclass ( x , BaseClasses . MOUNT )
2023-11-13 11:05:05 -05:00
)
)
)
2023-03-03 15:23:46 +00:00
{
2023-10-10 11:03:20 +00:00
// Add mod to allowed list
2023-03-03 15:23:46 +00:00
filteredScopesAndMods . push ( item ) ;
}
}
}
2023-11-13 11:05:05 -05:00
// No mods added to return list after filtering has occurred, send back the original mod list
2023-03-03 15:23:46 +00:00
if ( ! filteredScopesAndMods || filteredScopesAndMods . length === 0 )
{
2023-11-13 11:05:05 -05:00
this . logger . debug (
` Scope whitelist too restrictive for: ${ weapon . _tpl } ${ weaponDetails [ 1 ] . _name } , skipping filter ` ,
) ;
2023-03-03 15:23:46 +00:00
return scopes ;
}
return filteredScopesAndMods ;
}
2023-11-13 11:05:05 -05:00
}