import { inject, injectable } from "tsyringe"; 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"; import { PresetHelper } from "@spt-aki/helpers/PresetHelper"; import { ProbabilityHelper } from "@spt-aki/helpers/ProbabilityHelper"; import { ProfileHelper } from "@spt-aki/helpers/ProfileHelper"; import { WeightedRandomHelper } from "@spt-aki/helpers/WeightedRandomHelper"; 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"; import { ModSpawn } from "@spt-aki/models/enums/ModSpawn"; import { IChooseRandomCompatibleModResult } from "@spt-aki/models/spt/bots/IChooseRandomCompatibleModResult"; import { EquipmentFilterDetails, IBotConfig } from "@spt-aki/models/spt/config/IBotConfig"; 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"; import { IGenerateEquipmentProperties } from "./BotInventoryGenerator"; import { IFilterPlateModsForSlotByLevelResult, Result } from "./IFilterPlateModsForSlotByLevelResult"; @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, @inject("WeightedRandomHelper") protected weightedRandomHelper: WeightedRandomHelper, @inject("PresetHelper") protected presetHelper: PresetHelper, @inject("LocalisationService") protected localisationService: LocalisationService, @inject("BotEquipmentModPoolService") protected botEquipmentModPoolService: BotEquipmentModPoolService, @inject("ConfigServer") protected configServer: ConfigServer, ) { this.botConfig = this.configServer.getConfig(ConfigTypes.BOT); } /** * 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 */ public generateModsForEquipment( equipment: Item[], parentId: string, parentTemplate: ITemplateItem, settings: IGenerateEquipmentProperties, forceSpawn = false, ): Item[] { const compatibleModsPool = settings.modPool[parentTemplate._id]; if (!compatibleModsPool) { this.logger.warning( `bot: ${settings.botRole} lacks a mod slot pool for item: ${parentTemplate._id} ${parentTemplate._name}`, ); } // Iterate over mod pool and choose mods to add to item for (const modSlotName in compatibleModsPool) { const itemSlotTemplate = this.getModItemSlotFromDb(modSlotName, parentTemplate); if (!itemSlotTemplate) { this.logger.error( this.localisationService.getText("bot-mod_slot_missing_from_item", { modSlot: modSlotName, parentId: parentTemplate._id, parentName: parentTemplate._name, botRole: settings.botRole, }), ); continue; } const modSpawnResult = this.shouldModBeSpawned( itemSlotTemplate, modSlotName.toLowerCase(), settings.spawnChances.equipmentMods, ); if (modSpawnResult === ModSpawn.SKIP && !forceSpawn) { continue; } // Ensure submods for nvgs all spawn together if (modSlotName === "mod_nvg") { forceSpawn = true; } let modPoolToChooseFrom = compatibleModsPool[modSlotName]; if ( settings.botEquipmentConfig.filterPlatesByLevel && this.itemHelper.isRemovablePlateSlot(modSlotName.toLowerCase()) ) { const outcome = this.filterPlateModsForSlotByLevel( settings, modSlotName.toLowerCase(), compatibleModsPool[modSlotName], parentTemplate, ); if ([Result.UNKNOWN_FAILURE, Result.NO_DEFAULT_FILTER].includes(outcome.result)) { this.logger.debug( `Plate slot: ${modSlotName} selection for armor: ${parentTemplate._id} failed: ${ Result[outcome.result] }, skipping`, ); continue; } if ([Result.LACKS_PLATE_WEIGHTS].includes(outcome.result)) { this.logger.warning( `Plate slot: ${modSlotName} lacks weights for armor: ${parentTemplate._id}, unable to adjust plate choice, using existing data`, ); } modPoolToChooseFrom = outcome.plateModTpls; } // Find random mod and check its compatible let modTpl: string; let found = false; const exhaustableModPool = new ExhaustableArray(modPoolToChooseFrom, this.randomUtil, this.jsonUtil); while (exhaustableModPool.hasValues()) { modTpl = exhaustableModPool.getRandomValue(); if ( !this.botGeneratorHelper.isItemIncompatibleWithCurrentItems(equipment, modTpl, modSlotName) .incompatible ) { found = true; break; } } // Compatible item not found but slot REQUIRES item, get random item from db if (!found && itemSlotTemplate._required) { modTpl = this.getRandomModTplFromItemDb(modTpl, itemSlotTemplate, modSlotName, equipment); found = !!modTpl; } // Compatible item not found + not required if (!(found || itemSlotTemplate._required)) { // Don't add item continue; } const modTemplate = this.itemHelper.getItem(modTpl); if (!this.isModValidForSlot(modTemplate, itemSlotTemplate, modSlotName, parentTemplate, settings.botRole)) { continue; } // Generate new id to ensure all items are unique on bot const modId = this.hashUtil.generate(); equipment.push(this.createModItem(modId, modTpl, parentId, modSlotName, modTemplate[1], settings.botRole)); // Does the item being added have possible child mods? if (Object.keys(settings.modPool).includes(modTpl)) { // Call self recursively with item being checkced item we just added to bot this.generateModsForEquipment(equipment, modId, modTemplate[1], settings, forceSpawn); } } return equipment; } /** * Filter a bots plate pool based on its current level * @param settings Bot equipment generation settings * @param modSlot Armor slot being filtered * @param existingPlateTplPool Plates tpls to choose from * @param armorItem * @returns Array of plate tpls to choose from */ protected filterPlateModsForSlotByLevel( settings: IGenerateEquipmentProperties, modSlot: string, existingPlateTplPool: string[], armorItem: ITemplateItem, ): IFilterPlateModsForSlotByLevelResult { const result: IFilterPlateModsForSlotByLevelResult = { result: Result.UNKNOWN_FAILURE, plateModTpls: null }; // Not pmc or not a plate slot, return original mod pool array if (!this.itemHelper.isRemovablePlateSlot(modSlot)) { result.result = Result.NOT_PLATE_HOLDING_SLOT; result.plateModTpls = existingPlateTplPool; return result; } // Get the front/back/side weights based on bots level const plateSlotWeights = settings.botEquipmentConfig?.armorPlateWeighting?.find((x) => settings.botLevel >= x.levelRange.min && settings.botLevel <= x.levelRange.max ); if (!plateSlotWeights) { // No weights, return original array of plate tpls result.result = Result.LACKS_PLATE_WEIGHTS; result.plateModTpls = existingPlateTplPool; return result; } // Get the specific plate slot weights (front/back/side) const plateWeights: Record = plateSlotWeights[modSlot]; if (!plateWeights) { // No weights, return original array of plate tpls result.result = Result.LACKS_PLATE_WEIGHTS; result.plateModTpls = existingPlateTplPool; return result; } // Choose a plate level based on weighting const chosenArmorPlateLevel = this.weightedRandomHelper.getWeightedValue(plateWeights); // Convert the array of ids into database items const platesFromDb = existingPlateTplPool.map((x) => this.itemHelper.getItem(x)[1]); // Filter plates to the chosen level based on its armorClass property const filteredPlates = platesFromDb.filter((x) => x._props.armorClass === chosenArmorPlateLevel); if (filteredPlates.length === 0) { this.logger.debug( `Plate filter was too restrictive for armor: ${armorItem._id}, unable to find plates of level: ${chosenArmorPlateLevel}. Using mod items default plate`, ); const relatedItemDbModSlot = armorItem._props.Slots.find((slot) => slot._name.toLowerCase() === modSlot); const defaultPlate = relatedItemDbModSlot._props.filters[0].Plate; if (!defaultPlate) { // 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) { const relatedPresetSlot = defaultPreset._items.find((item) => item.slotId?.toLowerCase() === modSlot ); if (relatedPresetSlot) { result.result = Result.SUCCESS; result.plateModTpls = [relatedPresetSlot._tpl]; return result; } } result.result = Result.NO_DEFAULT_FILTER; return result; } result.result = Result.SUCCESS; result.plateModTpls = [defaultPlate]; return result; } // Only return the items ids result.result = Result.SUCCESS; result.plateModTpls = filteredPlates.map((x) => x._id); return result; } /** * 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 * @param weaponId parentId of weapon * @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 * @param botLevel Level of the bot weapon is being generated for * @param modLimits limits placed on certain mod types per gun * @param botEquipmentRole role of bot when accessing bot.json equipment config settings * @returns Weapon + mods array */ public generateModsForWeapon( sessionId: string, weapon: Item[], modPool: Mods, weaponId: string, parentTemplate: ITemplateItem, modSpawnChances: ModsChances, ammoTpl: string, botRole: string, botLevel: number, modLimits: BotModLimits, botEquipmentRole: string, ): Item[] { const pmcProfile = this.profileHelper.getPmcProfile(sessionId); // Get pool of mods that fit weapon const compatibleModsPool = modPool[parentTemplate._id]; if ( !((parentTemplate._props.Slots.length || parentTemplate._props.Cartridges?.length) || parentTemplate._props.Chambers?.length) ) { this.logger.error( this.localisationService.getText("bot-unable_to_add_mods_to_weapon_missing_ammo_slot", { weaponName: parentTemplate._name, weaponId: parentTemplate._id, botRole: botRole, }), ); return weapon; } const botEquipConfig = this.botConfig.equipment[botEquipmentRole]; const botEquipBlacklist = this.botEquipmentFilterService.getBotEquipmentBlacklist( botEquipmentRole, pmcProfile.Info.Level, ); const botWeaponSightWhitelist = this.botEquipmentFilterService.getBotWeaponSightWhitelist(botEquipmentRole); const randomisationSettings = this.botHelper.getBotRandomizationDetails(botLevel, botEquipConfig); // Iterate over mod pool and choose mods to attach const sortedModKeys = this.sortModKeys(Object.keys(compatibleModsPool)); for (const modSlot of sortedModKeys) { // Check weapon has slot for mod to fit in const modsParentSlot = this.getModItemSlotFromDb(modSlot, parentTemplate); if (!modsParentSlot) { this.logger.error( this.localisationService.getText("bot-weapon_missing_mod_slot", { modSlot: modSlot, weaponId: parentTemplate._id, weaponName: parentTemplate._name, botRole: botRole, }), ); continue; } // Check spawn chance of mod const modSpawnResult = this.shouldModBeSpawned(modsParentSlot, modSlot, modSpawnChances); if (modSpawnResult === ModSpawn.SKIP) { continue; } const isRandomisableSlot = randomisationSettings?.randomisedWeaponModSlots?.includes(modSlot) ?? false; const modToAdd = this.chooseModToPutIntoSlot( modSlot, isRandomisableSlot, botWeaponSightWhitelist, botEquipBlacklist, compatibleModsPool, weapon, ammoTpl, parentTemplate, modSpawnResult, ); // Compatible mod not found if (!modToAdd || typeof modToAdd === "undefined") { continue; } if (!this.isModValidForSlot(modToAdd, modsParentSlot, modSlot, parentTemplate, botRole)) { continue; } const modToAddTemplate = modToAdd[1]; // Skip adding mod to weapon if type limit reached if ( this.botWeaponModLimitService.weaponModHasReachedLimit( botEquipmentRole, modToAddTemplate, modLimits, parentTemplate, weapon, ) ) { 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 const scopeSlots = ["mod_scope", "mod_scope_000", "mod_scope_001", "mod_scope_002", "mod_scope_003"]; this.adjustSlotSpawnChances(modSpawnChances, scopeSlots, 100); // 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); } } // If picked item is muzzle adapter that can hold a child, adjust spawn chance if (this.modSlotCanHoldMuzzleDevices(modSlot, modToAddTemplate._parent)) { const muzzleSlots = ["mod_muzzle", "mod_muzzle_000", "mod_muzzle_001"]; // Make chance of muzzle devices 95%, nearly certain but not guaranteed this.adjustSlotSpawnChances(modSpawnChances, muzzleSlots, 95); } // If front/rear sight are to be added, set opposite to 100% chance if (this.modIsFrontOrRearSight(modSlot, modToAddTemplate._id)) { modSpawnChances.mod_sight_front = 100; modSpawnChances.mod_sight_rear = 100; } // 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 if ( modSlot === "mod_handguard" && modToAddTemplate._props.Slots.find((x) => x._name === "mod_handguard") && !weapon.find((x) => x.slotId === "mod_launcher") ) { // Needed for handguards with lower modSpawnChances.mod_handguard = 100; } // 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 if ( modSlot === "mod_stock" && (modToAddTemplate._props.Slots.find((x) => x._name.includes("mod_stock") || botEquipConfig.forceStock )) ) { // Stock mod can take additional stocks, could be a locking device, force 100% chance const stockSlots = ["mod_stock", "mod_stock_000", "mod_stock_akms"]; this.adjustSlotSpawnChances(modSpawnChances, stockSlots, 100); } const modId = this.hashUtil.generate(); weapon.push(this.createModItem(modId, modToAddTemplate._id, weaponId, modSlot, modToAddTemplate, botRole)); // I first thought we could use the recursive generateModsForItems as previously for cylinder magazines. // However, the recursion doesn't go over the slots of the parent mod but over the modPool which is given by the bot config // 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) { // Call self recursively to add mods to this mod this.generateModsForWeapon( sessionId, weapon, modPool, modId, modToAddTemplate, modSpawnChances, ammoTpl, botRole, botLevel, modLimits, botEquipmentRole, ); } } } return weapon; } /** * Is this modslot a front or rear sight * @param modSlot Slot to check * @returns true if it's a front/rear sight */ protected modIsFrontOrRearSight(modSlot: string, tpl: string): boolean { if (modSlot === "mod_gas_block" && tpl === "5ae30e795acfc408fb139a0b") { // M4A1 front sight with gas block return true; } 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 { return [ "mod_scope", "mod_mount", "mod_mount_000", "mod_scope_000", "mod_scope_001", "mod_scope_002", "mod_scope_003", ].includes(modSlot.toLowerCase()) && modsParentId === BaseClasses.MOUNT; } /** * Set mod spawn chances to defined amount * @param modSpawnChances Chance dictionary to update */ protected adjustSlotSpawnChances( modSpawnChances: ModsChances, modSlotsToAdjust: string[], newChancePercent: number, ): void { if (!modSpawnChances) { this.logger.warning("Unable to adjust scope spawn chances as spawn chance object is empty"); return; } if (!modSlotsToAdjust) { return; } for (const modName of modSlotsToAdjust) { modSpawnChances[modName] = newChancePercent; } } protected modSlotCanHoldMuzzleDevices(modSlot: string, modsParentId: string): boolean { return ["mod_muzzle", "mod_muzzle_000", "mod_muzzle_001"].includes(modSlot.toLowerCase()); } protected sortModKeys(unsortedKeys: string[]): string[] { if (unsortedKeys.length <= 1) { return unsortedKeys; } const sortedKeys: string[] = []; const modRecieverKey = "mod_reciever"; const modMount001Key = "mod_mount_001"; const modGasBlockKey = "mod_gas_block"; const modPistolGrip = "mod_pistol_grip"; const modStockKey = "mod_stock"; const modBarrelKey = "mod_barrel"; const modHandguardKey = "mod_handguard"; const modMountKey = "mod_mount"; const modScopeKey = "mod_scope"; if (unsortedKeys.includes(modHandguardKey)) { sortedKeys.push(modHandguardKey); unsortedKeys.splice(unsortedKeys.indexOf(modHandguardKey), 1); } 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); } if (unsortedKeys.includes(modPistolGrip)) { sortedKeys.push(modPistolGrip); unsortedKeys.splice(unsortedKeys.indexOf(modPistolGrip), 1); } if (unsortedKeys.includes(modGasBlockKey)) { sortedKeys.push(modGasBlockKey); unsortedKeys.splice(unsortedKeys.indexOf(modGasBlockKey), 1); } 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 */ protected getModItemSlotFromDb(modSlot: string, parentTemplate: ITemplateItem): Slot { const modSlotLower = modSlot.toLowerCase(); switch (modSlotLower) { case "patron_in_weapon": case "patron_in_weapon_000": case "patron_in_weapon_001": return parentTemplate._props.Chambers.find((chamber) => chamber._name.includes(modSlotLower)); case "cartridges": return parentTemplate._props.Cartridges.find((c) => c._name.toLowerCase() === modSlotLower); default: return parentTemplate._props.Slots.find((s) => s._name.toLowerCase() === modSlotLower); } } /** * Randomly choose if a mod should be spawned, 100% for required mods OR mod is ammo slot * @param itemSlot slot the item sits in * @param modSlot slot the mod sits in * @param modSpawnChances Chances for various mod spawns * @returns ModSpawn.SPAWN when mod should be spawned, ModSpawn.DEFAULT_MOD when default mod should spawn, ModSpawn.SKIP when mod is skipped */ protected shouldModBeSpawned(itemSlot: Slot, modSlot: string, modSpawnChances: ModsChances): ModSpawn { const slotRequired = itemSlot._required; if (this.getAmmoContainers().includes(modSlot)) { return ModSpawn.SPAWN; } const spawnMod = this.probabilityHelper.rollChance(modSpawnChances[modSlot]); if (!spawnMod && slotRequired) { // Mod is required but spawn chance roll failed, choose default mod spawn for slot return ModSpawn.DEFAULT_MOD; } return spawnMod ? ModSpawn.SPAWN : ModSpawn.SKIP; } /** * @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 * @returns itemHelper.getItem() result */ protected chooseModToPutIntoSlot( modSlot: string, isRandomisableSlot: boolean, botWeaponSightWhitelist: Record, botEquipBlacklist: EquipmentFilterDetails, itemModPool: Record, weapon: Item[], ammoTpl: string, parentTemplate: ITemplateItem, modSpawnResult: ModSpawn, ): [boolean, ITemplateItem] { /** Slot mod will fill */ const parentSlot = parentTemplate._props.Slots.find((i) => i._name === modSlot); const weaponTemplate = this.itemHelper.getItem(weapon[0]._tpl)[1]; // It's ammo, use predefined ammo parameter if (this.getAmmoContainers().includes(modSlot) && modSlot !== "mod_magazine") { return this.itemHelper.getItem(ammoTpl); } // Ensure there's a pool of mods to pick from let modPool = this.getModPoolForSlot( itemModPool, modSpawnResult, parentTemplate, weaponTemplate, modSlot, botEquipBlacklist, isRandomisableSlot, ); if (!(modPool || parentSlot._required)) { // Nothing in mod pool + item not required this.logger.debug(`Mod pool for slot: ${modSlot} on item: ${parentTemplate._name} was empty, skipping mod`); return null; } // 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) { modPool = this.filterSightsByWeaponType(weapon[0], modPool, botWeaponSightWhitelist); } } // Pick random mod that's compatible const chosenModResult = this.pickWeaponModTplForSlotFromPool( modPool, parentSlot, modSpawnResult, weapon, modSlot, ); if (chosenModResult.slotBlocked && !parentSlot._required) { // 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); // this.logger.debug(`Weapon: ${weapon.map(x => `${x._tpl} ${x.slotId ?? ""}`).join(",")}`) } // Get random mod to attach from items db for required slots if none found above if (!chosenModResult.found && parentSlot !== undefined && parentSlot._required) { chosenModResult.chosenTpl = this.getRandomModTplFromItemDb("", parentSlot, modSlot, weapon); chosenModResult.found = true; } // Compatible item not found + not required if (!chosenModResult.found && parentSlot !== undefined && !parentSlot._required) { return null; } if (!chosenModResult.found && parentSlot !== undefined) { if (parentSlot._required) { this.logger.warning( `Required slot unable to be filled, ${modSlot} on ${parentTemplate._name} ${parentTemplate._id} for weapon: ${ weapon[0]._tpl }`, ); } return null; } return this.itemHelper.getItem(chosenModResult.chosenTpl); } protected pickWeaponModTplForSlotFromPool( modPool: string[], parentSlot: Slot, modSpawnResult: ModSpawn, weapon: Item[], modSlotname: string, ): IChooseRandomCompatibleModResult { let chosenTpl: string; const exhaustableModPool = new ExhaustableArray(modPool, this.randomUtil, this.jsonUtil); let chosenModResult: IChooseRandomCompatibleModResult = { incompatible: true, found: false, reason: "unknown" }; const modParentFilterList = parentSlot._props.filters[0].Filter; // 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 let blockedAttemptCount = 0; 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; chosenModResult.incompatible = false; chosenModResult.chosenTpl = chosenTpl; break; } // Check chosen item is on the allowed list of the parent const isOnModParentFilterList = modParentFilterList.includes(chosenTpl); if (!isOnModParentFilterList) { // Try again continue; } chosenModResult = this.botGeneratorHelper.isWeaponModIncompatibleWithCurrentMods( weapon, chosenTpl, modSlotname, ); if (chosenModResult.slotBlocked) { // Give max of x attempts of picking a mod if blocked by another if (blockedAttemptCount > maxBlockedAttempts) { blockedAttemptCount = 0; break; } blockedAttemptCount++; // Try again continue; } // Some mod combos will never work, make sure this isnt the case if (!(chosenModResult.incompatible || this.weaponModComboIsIncompatible(weapon, chosenTpl))) { // Success chosenModResult.found = true; chosenModResult.incompatible = false; chosenModResult.chosenTpl = chosenTpl; break; } } return chosenModResult; } /** * 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 * @param botEquipBlacklist * @param isRandomisableSlot is flagged as a randomisable slot * @returns */ protected getModPoolForSlot( itemModPool: Record, modSpawnResult: ModSpawn, parentTemplate: ITemplateItem, weaponTemplate: ITemplateItem, modSlot: string, botEquipBlacklist: EquipmentFilterDetails, isRandomisableSlot: boolean, ): string[] { // Mod is flagged as being default only, try and find it in globals if (modSpawnResult === ModSpawn.DEFAULT_MOD) { // Edge case - using mp5sd reciever means default mp5 handguard doesnt fit const isMp5sd = parentTemplate._id === "5926f2e086f7745aae644231"; const defaultWeaponPreset = isMp5sd ? this.presetHelper.getPreset("59411abb86f77478f702b5d2") : this.presetHelper.getDefaultPreset(weaponTemplate._id); const matchingMod = defaultWeaponPreset._items.find((item) => item?.slotId?.toLowerCase() === modSlot.toLowerCase() ); // 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 if (matchingMod) { // 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]; } } this.logger.debug(`No default: ${modSlot} mod found on template: ${weaponTemplate._id}`); // Couldnt find default in globals, use existing mod pool data 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]; } /** * 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; } /** * Create a mod item with parameters as properties * @param modId _id * @param modTpl _tpl * @param parentId parentId * @param modSlot slotId * @param modTemplate Used to add additional properties in the upd object * @returns Item object */ protected createModItem( modId: string, modTpl: string, parentId: string, modSlot: string, modTemplate: ITemplateItem, botRole: string, ): Item { return { _id: modId, _tpl: modTpl, parentId: parentId, slotId: modSlot, ...this.botGeneratorHelper.generateExtraPropertiesForItem(modTemplate, botRole), }; } /** * 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 * @param parentSlot item mod will go into, used to get compatible items * @param modSlot Slot to get mod to fill * @param items items to ensure picked mod is compatible with * @returns item tpl */ protected getRandomModTplFromItemDb(modTpl: string, parentSlot: Slot, modSlot: string, items: Item[]): string { // Find compatible mods and make an array of them 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 * @param slotAddedToTemplate slot the item will be placed in * @param modSlot slot the mod will fill * @param parentTemplate template of the mods being added * @param botRole * @returns true if valid */ protected isModValidForSlot( modToAdd: [boolean, ITemplateItem], slotAddedToTemplate: Slot, modSlot: string, parentTemplate: ITemplateItem, botRole: string, ): boolean { const modBeingAddedTemplate = modToAdd[1]; // Mod lacks template item if (!modBeingAddedTemplate) { this.logger.error( this.localisationService.getText("bot-no_item_template_found_when_adding_mod", { modId: modBeingAddedTemplate._id, modSlot: modSlot, }), ); 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 if (slotAddedToTemplate._required) { this.logger.warning( this.localisationService.getText("bot-unable_to_add_mod_item_invalid", { itemName: modBeingAddedTemplate._name, modSlot: modSlot, parentItemName: parentTemplate._name, botRole: botRole, }), ); } 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 */ protected addCompatibleModsForProvidedMod( desiredSlotName: string, modTemplate: ITemplateItem, modPool: Mods, botEquipBlacklist: EquipmentFilterDetails, ): void { const desiredSlotObject = modTemplate._props.Slots.find((x) => x._name.includes(desiredSlotName)); if (desiredSlotObject) { const supportedSubMods = desiredSlotObject._props.filters[0].Filter; if (supportedSubMods) { // Filter mods let filteredMods = this.filterWeaponModsByBlacklist( supportedSubMods, botEquipBlacklist, desiredSlotName, ); if (filteredMods.length === 0) { this.logger.warning( this.localisationService.getText("bot-unable_to_filter_mods_all_blacklisted", { slotName: desiredSlotObject._name, itemName: modTemplate._name, }), ); 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 */ protected getDynamicModPool( parentItemId: string, modSlot: string, botEquipBlacklist: EquipmentFilterDetails, ): string[] { const modsFromDynamicPool = this.jsonUtil.clone( this.botEquipmentModPoolService.getCompatibleModsForWeaponSlot(parentItemId, modSlot), ); const filteredMods = this.filterWeaponModsByBlacklist(modsFromDynamicPool, botEquipBlacklist, modSlot); if (filteredMods.length === 0) { this.logger.warning( this.localisationService.getText("bot-unable_to_filter_mod_slot_all_blacklisted", modSlot), ); 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 */ protected filterWeaponModsByBlacklist( allowedMods: string[], botEquipBlacklist: EquipmentFilterDetails, modSlot: string, ): string[] { if (!botEquipBlacklist) { return allowedMods; } let result: string[] = []; // 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)); return result; } /** * With the shotgun revolver (60db29ce99594040e04c4a27) 12.12 introduced CylinderMagazines. * Those magazines (e.g. 60dc519adf4c47305f6d410d) 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 * @param modPool modPool which should include available cartridges * @param parentId The CylinderMagazine's UID * @param parentTemplate The CylinderMagazine's template */ protected fillCamora(items: Item[], modPool: Mods, parentId: string, parentTemplate: ITemplateItem): void { let itemModPool = modPool[parentTemplate._id]; if (!itemModPool) { 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")); // 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]; } 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; exhaustableModPool = new ExhaustableArray( this.mergeCamoraPoolsTogether(itemModPool), this.randomUtil, this.jsonUtil, ); } 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(); items.push({ _id: modId, _tpl: modTpl, parentId: parentId, slotId: modSlotId }); } } /** * Take a record of camoras and merge the compatible shells into one array * @param camorasWithShells camoras we want to merge into one array * @returns string array of shells for multiple camora sources */ protected mergeCamoraPoolsTogether(camorasWithShells: Record): string[] { 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 * Controlled by bot.json weaponSightWhitelist * e.g. filter out rifle scopes from SMGs * @param weapon Weapon scopes will be added to * @param scopes Full scope pool * @param botWeaponSightWhitelist Whitelist of scope types by weapon base type * @returns Array of scope tpls that have been filtered to just ones allowed for that weapon type */ protected filterSightsByWeaponType( weapon: Item, scopes: string[], botWeaponSightWhitelist: Record, ): string[] { const weaponDetails = this.itemHelper.getItem(weapon._tpl); // Return original scopes array if whitelist not found const whitelistedSightTypes = botWeaponSightWhitelist[weaponDetails[1]._parent]; if (!whitelistedSightTypes) { this.logger.debug( `Unable to find whitelist for weapon type: ${weaponDetails[1]._parent} ${ weaponDetails[1]._name }, skipping sight filtering`, ); 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)) { // Add mod to allowed list filteredScopesAndMods.push(item); continue; } // Edge case, what if item is a mount for a scope and not directly a scope? // Check item is mount + has child items const itemDetails = this.itemHelper.getItem(item)[1]; if (this.itemHelper.isOfBaseclass(item, BaseClasses.MOUNT) && itemDetails._props.Slots.length > 0) { // 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 const scopeSlot = itemDetails._props.Slots.filter((x) => ["mod_scope", "mod_scope_000"].includes(x._name) ); // Mods scope slot found must allow ALL whitelisted scope types OR be a mount if ( scopeSlot?.every((x) => x._props.filters[0].Filter.every((x) => this.itemHelper.isOfBaseclasses(x, whitelistedSightTypes) || this.itemHelper.isOfBaseclass(x, BaseClasses.MOUNT) ) ) ) { // Add mod to allowed list filteredScopesAndMods.push(item); } } } // No mods added to return list after filtering has occurred, send back the original mod list if (!filteredScopesAndMods || filteredScopesAndMods.length === 0) { this.logger.debug( `Scope whitelist too restrictive for: ${weapon._tpl} ${weaponDetails[1]._name}, skipping filter`, ); return scopes; } return filteredScopesAndMods; } }