import { inject, injectable } from "tsyringe"; import { DurabilityLimitsHelper } from "../helpers/DurabilityLimitsHelper"; import { Item, Repairable, Upd } from "../models/eft/common/tables/IItem"; import { ITemplateItem } from "../models/eft/common/tables/ITemplateItem"; import { BaseClasses } from "../models/enums/BaseClasses"; import { ConfigTypes } from "../models/enums/ConfigTypes"; import { EquipmentFilters, IBotConfig } from "../models/spt/config/IBotConfig"; import { ILogger } from "../models/spt/utils/ILogger"; import { ConfigServer } from "../servers/ConfigServer"; import { DatabaseServer } from "../servers/DatabaseServer"; import { LocalisationService } from "../services/LocalisationService"; import { JsonUtil } from "../utils/JsonUtil"; import { RandomUtil } from "../utils/RandomUtil"; import { ItemHelper } from "./ItemHelper"; @injectable() export class BotGeneratorHelper { protected botConfig: IBotConfig; constructor( @inject("WinstonLogger") protected logger: ILogger, @inject("RandomUtil") protected randomUtil: RandomUtil, @inject("DatabaseServer") protected databaseServer: DatabaseServer, @inject("DurabilityLimitsHelper") protected durabilityLimitsHelper: DurabilityLimitsHelper, @inject("ItemHelper") protected itemHelper: ItemHelper, @inject("LocalisationService") protected localisationService: LocalisationService, @inject("ConfigServer") protected configServer: ConfigServer ) { this.botConfig = this.configServer.getConfig(ConfigTypes.BOT); } /** * Adds properties to an item * e.g. Repairable / HasHinge / Foldable / MaxDurability * @param itemTemplate Item extra properties are being generated for * @param botRole Used by weapons to randomize the durability values. Null for non-equipped items * @returns Item Upd object with extra properties */ public generateExtraPropertiesForItem(itemTemplate: ITemplateItem, botRole: string = null): { upd?: Upd } { const properties: Upd = {}; if (itemTemplate._props.MaxDurability) { if (itemTemplate._props.weapClass) // Is weapon { properties.Repairable = this.generateWeaponRepairableProperties(itemTemplate, botRole); } else if (itemTemplate._props.armorClass) // Is armor { properties.Repairable = this.generateArmorRepairableProperties(itemTemplate, botRole); } } if (itemTemplate._props.HasHinge) { properties.Togglable = { On: true }; } if (itemTemplate._props.Foldable) { properties.Foldable = { Folded: false }; } if (itemTemplate._props.weapFireType?.length) { if (itemTemplate._props.weapFireType.includes("fullauto")) { properties.FireMode = { FireMode: "fullauto" }; } else { properties.FireMode = { FireMode: this.randomUtil.getArrayValue(itemTemplate._props.weapFireType) }; } } if (itemTemplate._props.MaxHpResource) { properties.MedKit = { HpResource: itemTemplate._props.MaxHpResource }; } if (itemTemplate._props.MaxResource && itemTemplate._props.foodUseTime) { properties.FoodDrink = { HpPercent: itemTemplate._props.MaxResource }; } if ([BaseClasses.FLASHLIGHT, BaseClasses.TACTICAL_COMBO].includes(itemTemplate._parent)) { // Get chance from botconfig for bot type, use 50% if no value found const lightLaserActiveChance = this.getBotEquipmentSettingFromConfig(botRole, "lightLaserIsActiveChancePercent", 50); properties.Light = { IsActive: (this.randomUtil.getChance100(lightLaserActiveChance)), SelectedMode: 0 }; } if (itemTemplate._parent === BaseClasses.NIGHTVISION) { // Get chance from botconfig for bot type, use 50% if no value found const nvgActiveChance = this.getBotEquipmentSettingFromConfig(botRole, "nvgIsActiveChancePercent", 50); properties.Togglable = { On: (this.randomUtil.getChance100(nvgActiveChance)) }; } // Togglable face shield if (itemTemplate._props.HasHinge && itemTemplate._props.FaceShieldComponent) { // Get chance from botconfig for bot type, use 75% if no value found const faceShieldActiveChance = this.getBotEquipmentSettingFromConfig(botRole, "faceShieldIsActiveChancePercent", 75); properties.Togglable = { On: (this.randomUtil.getChance100(faceShieldActiveChance)) }; } return Object.keys(properties).length ? { upd: properties } : {}; } /** * Get the chance for the weapon attachment or helmet equipment to be set as activated * @param botRole role of bot with weapon/helmet * @param setting the setting of the weapon attachment/helmet equipment to be activated * @param defaultValue default value for the chance of activation if the botrole or bot equipment role is null * @returns Percent chance to be active */ protected getBotEquipmentSettingFromConfig(botRole: string, setting: keyof EquipmentFilters, defaultValue: number): number { if (!botRole) { return defaultValue; } const botEquipmentSettings = this.botConfig.equipment[this.getBotEquipmentRole(botRole)]; if (!botEquipmentSettings) { this.logger.warning(this.localisationService.getText("bot-missing_equipment_settings", {botRole: botRole, setting: setting, defaultValue: defaultValue})); return defaultValue; } if (botEquipmentSettings[setting] === undefined || typeof botEquipmentSettings[setting] !== "number") { this.logger.warning(this.localisationService.getText("bot-missing_equipment_settings_property", {botRole: botRole, setting: setting, defaultValue: defaultValue})); return defaultValue; } return botEquipmentSettings[setting]; } /** * Create a repairable object for a weapon that containers durability + max durability properties * @param itemTemplate weapon object being generated for * @param botRole type of bot being generated for * @returns Repairable object */ protected generateWeaponRepairableProperties(itemTemplate: ITemplateItem, botRole: string): Repairable { const maxDurability = this.durabilityLimitsHelper.getRandomizedMaxWeaponDurability(itemTemplate, botRole); const currentDurability = this.durabilityLimitsHelper.getRandomizedWeaponDurability(itemTemplate, botRole, maxDurability); return { Durability: currentDurability, MaxDurability: maxDurability }; } /** * Create a repairable object for an armor that containers durability + max durability properties * @param itemTemplate weapon object being generated for * @param botRole type of bot being generated for * @returns Repairable object */ protected generateArmorRepairableProperties(itemTemplate: ITemplateItem, botRole: string): Repairable { let maxDurability: number; let currentDurability: number; if (parseInt(`${itemTemplate._props.armorClass}`) === 0) { maxDurability = itemTemplate._props.MaxDurability; currentDurability = itemTemplate._props.MaxDurability; } else { maxDurability = this.durabilityLimitsHelper.getRandomizedMaxArmorDurability(itemTemplate, botRole); currentDurability = this.durabilityLimitsHelper.getRandomizedArmorDurability(itemTemplate, botRole, maxDurability); } return { Durability: currentDurability, MaxDurability: maxDurability }; } /** * Can item be added to another item without conflict * @param items Items to check compatibilities with * @param tplToCheck Tpl of the item to check for incompatibilities * @param equipmentSlot Slot the item will be placed into * @returns false if no incompatibilities, also has incompatibility reason */ public isItemIncompatibleWithCurrentItems(items: Item[], tplToCheck: string, equipmentSlot: string): { incompatible: boolean, reason: string } { // Skip slots that have no incompatibilities if (["Scabbard", "Backpack", "SecureContainer", "Holster", "ArmBand"].includes(equipmentSlot)) { return { incompatible: false, reason: "" }; } // TODO: Can probably be optimized to cache itemTemplates as items are added to inventory const equippedItems = items.map(i => this.databaseServer.getTables().templates.items[i._tpl]); const itemToEquip = this.itemHelper.getItem(tplToCheck); if (!itemToEquip[0]) { this.logger.warning(this.localisationService.getText("bot-invalid_item_compatibility_check", {itemTpl: tplToCheck, slot: equipmentSlot})); } if (!itemToEquip[1]._props) { this.logger.warning(this.localisationService.getText("bot-compatibility_check_missing_props", {id: itemToEquip[1]._id, name: itemToEquip[1]._name, slot: equipmentSlot})); } // Does an equipped item have a property that blocks the desired item - check for prop "BlocksX" .e.g BlocksEarpiece / BlocksFaceCover let blockingItem = equippedItems.find(x => x._props[`Blocks${equipmentSlot}`]); if (blockingItem) { //this.logger.warning(`1 incompatibility found between - ${itemToEquip[1]._name} and ${blockingItem._name} - ${equipmentSlot}`); return { incompatible: true, reason: `${tplToCheck} ${itemToEquip[1]._name} in slot: ${equipmentSlot} blocked by: ${blockingItem._id} ${blockingItem._name}` }; } // Check if any of the current inventory templates have the incoming item defined as incompatible blockingItem = equippedItems.find(x => x._props.ConflictingItems.includes(tplToCheck)); if (blockingItem) { //this.logger.warning(`2 incompatibility found between - ${itemToEquip[1]._name} and ${blockingItem._props.Name} - ${equipmentSlot}`); return { incompatible: true, reason: `${tplToCheck} ${itemToEquip[1]._name} in slot: ${equipmentSlot} blocked by: ${blockingItem._id} ${blockingItem._name}` }; } // Check if the incoming item has any inventory items defined as incompatible const blockingInventoryItem = items.find(x => itemToEquip[1]._props[`Blocks${x.slotId}`] || itemToEquip[1]._props.ConflictingItems.includes(x._tpl)); if (blockingInventoryItem) { //this.logger.warning(`3 incompatibility found between - ${itemToEquip[1]._name} and ${blockingInventoryItem._tpl} - ${equipmentSlot}`) return { incompatible: true, reason: `${tplToCheck} blocks existing item ${blockingInventoryItem._tpl} in slot ${blockingInventoryItem.slotId}` }; } return { incompatible: false, reason: "" }; } /** * Convert a bots role to the equipment role used in config/bot.json * @param botRole Role to convert * @returns Equipment role (e.g. pmc / assault / bossTagilla) */ public getBotEquipmentRole(botRole: string): string { return ([this.botConfig.pmc.usecType.toLowerCase(), this.botConfig.pmc.bearType.toLowerCase()].includes(botRole.toLowerCase())) ? "pmc" : botRole; } } /** TODO - move into own class */ export class ExhaustableArray { private pool: T[]; constructor( private itemPool: T[], private randomUtil: RandomUtil, private jsonUtil: JsonUtil ) { this.pool = this.jsonUtil.clone(itemPool); } public getRandomValue(): T { if (!this.pool?.length) { return null; } const index = this.randomUtil.getInt(0, this.pool.length - 1); const toReturn = this.jsonUtil.clone(this.pool[index]); this.pool.splice(index, 1); return toReturn; } public getFirstValue(): T { if (!this.pool?.length) { return null; } const toReturn = this.jsonUtil.clone(this.pool[0]); this.pool.splice(0, 1); return toReturn; } public hasValues(): boolean { if (this.pool?.length) { return true; } return false; } }