2023-03-03 16:23:46 +01:00
|
|
|
import { inject, injectable } from "tsyringe";
|
2024-02-25 12:45:34 +01:00
|
|
|
import { BotGeneratorHelper } from "@spt-aki/helpers/BotGeneratorHelper";
|
2023-10-19 19:21:17 +02:00
|
|
|
import { ItemHelper } from "@spt-aki/helpers/ItemHelper";
|
|
|
|
import { WeightedRandomHelper } from "@spt-aki/helpers/WeightedRandomHelper";
|
|
|
|
import { Inventory } from "@spt-aki/models/eft/common/tables/IBotBase";
|
|
|
|
import { GenerationData } from "@spt-aki/models/eft/common/tables/IBotType";
|
|
|
|
import { Item } from "@spt-aki/models/eft/common/tables/IItem";
|
2024-02-25 12:45:34 +01:00
|
|
|
import { ITemplateItem } from "@spt-aki/models/eft/common/tables/ITemplateItem";
|
2023-10-19 19:21:17 +02:00
|
|
|
import { BaseClasses } from "@spt-aki/models/enums/BaseClasses";
|
|
|
|
import { EquipmentSlots } from "@spt-aki/models/enums/EquipmentSlots";
|
|
|
|
import { ItemAddedResult } from "@spt-aki/models/enums/ItemAddedResult";
|
|
|
|
import { ILogger } from "@spt-aki/models/spt/utils/ILogger";
|
|
|
|
import { DatabaseServer } from "@spt-aki/servers/DatabaseServer";
|
|
|
|
import { LocalisationService } from "@spt-aki/services/LocalisationService";
|
|
|
|
import { HashUtil } from "@spt-aki/utils/HashUtil";
|
|
|
|
import { RandomUtil } from "@spt-aki/utils/RandomUtil";
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
@injectable()
|
|
|
|
export class BotWeaponGeneratorHelper
|
|
|
|
{
|
|
|
|
constructor(
|
|
|
|
@inject("WinstonLogger") protected logger: ILogger,
|
|
|
|
@inject("DatabaseServer") protected databaseServer: DatabaseServer,
|
|
|
|
@inject("ItemHelper") protected itemHelper: ItemHelper,
|
|
|
|
@inject("RandomUtil") protected randomUtil: RandomUtil,
|
|
|
|
@inject("HashUtil") protected hashUtil: HashUtil,
|
2023-10-10 13:03:20 +02:00
|
|
|
@inject("WeightedRandomHelper") protected weightedRandomHelper: WeightedRandomHelper,
|
2024-02-25 12:45:34 +01:00
|
|
|
@inject("BotGeneratorHelper") protected botGeneratorHelper: BotGeneratorHelper,
|
2023-03-03 16:23:46 +01:00
|
|
|
@inject("LocalisationService") protected localisationService: LocalisationService,
|
|
|
|
)
|
2023-11-16 22:42:06 +01:00
|
|
|
{}
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a randomized number of bullets for a specific magazine
|
2023-10-10 13:03:20 +02:00
|
|
|
* @param magCounts Weights of magazines
|
2023-03-03 16:23:46 +01:00
|
|
|
* @param magTemplate magazine to generate bullet count for
|
|
|
|
* @returns bullet count number
|
|
|
|
*/
|
2023-10-10 13:03:20 +02:00
|
|
|
public getRandomizedBulletCount(magCounts: GenerationData, magTemplate: ITemplateItem): number
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
const randomizedMagazineCount = this.getRandomizedMagazineCount(magCounts);
|
|
|
|
const parentItem = this.itemHelper.getItem(magTemplate._parent)[1];
|
|
|
|
let chamberBulletCount = 0;
|
|
|
|
if (this.magazineIsCylinderRelated(parentItem._name))
|
|
|
|
{
|
2023-11-24 16:24:46 +01:00
|
|
|
const firstSlotAmmoTpl = magTemplate._props.Cartridges[0]._props.filters[0].Filter[0];
|
|
|
|
const ammoMaxStackSize = this.itemHelper.getItem(firstSlotAmmoTpl)[1]?._props?.StackMaxSize ?? 1;
|
2024-05-17 21:32:41 +02:00
|
|
|
chamberBulletCount
|
|
|
|
= ammoMaxStackSize === 1
|
|
|
|
? 1 // Rotating grenade launcher
|
|
|
|
: magTemplate._props.Slots.length; // Shotguns/revolvers. We count the number of camoras as the _max_count of the magazine is 0
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
else if (parentItem._id === BaseClasses.UBGL)
|
|
|
|
{
|
2023-11-24 16:24:46 +01:00
|
|
|
// Underbarrel launchers can only have 1 chambered grenade
|
2023-03-03 16:23:46 +01:00
|
|
|
chamberBulletCount = 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
chamberBulletCount = magTemplate._props.Cartridges[0]._max_count;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get the amount of bullets that would fit in the internal magazine
|
2024-05-17 21:32:41 +02:00
|
|
|
* and multiply by how many magazines were supposed to be created */
|
2023-11-16 22:42:06 +01:00
|
|
|
return chamberBulletCount * randomizedMagazineCount;
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a randomized count of magazines
|
|
|
|
* @param magCounts min and max value returned value can be between
|
|
|
|
* @returns numerical value of magazine count
|
|
|
|
*/
|
2023-10-10 13:03:20 +02:00
|
|
|
public getRandomizedMagazineCount(magCounts: GenerationData): number
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
2023-11-16 22:42:06 +01:00
|
|
|
// const range = magCounts.max - magCounts.min;
|
|
|
|
// return this.randomUtil.getBiasedRandomNumber(magCounts.min, magCounts.max, Math.round(range * 0.75), 4);
|
2023-10-10 13:03:20 +02:00
|
|
|
|
2023-11-24 16:31:16 +01:00
|
|
|
return Number.parseInt(this.weightedRandomHelper.getWeightedValue(magCounts.weights));
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Is this magazine cylinder related (revolvers and grenade launchers)
|
|
|
|
* @param magazineParentName the name of the magazines parent
|
|
|
|
* @returns true if it is cylinder related
|
|
|
|
*/
|
|
|
|
public magazineIsCylinderRelated(magazineParentName: string): boolean
|
|
|
|
{
|
|
|
|
return ["CylinderMagazine", "SpringDrivenCylinder"].includes(magazineParentName);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a magazine using the parameters given
|
|
|
|
* @param magazineTpl Tpl of the magazine to create
|
|
|
|
* @param ammoTpl Ammo to add to magazine
|
|
|
|
* @param magTemplate template object of magazine
|
|
|
|
* @returns Item array
|
|
|
|
*/
|
2023-10-10 13:03:20 +02:00
|
|
|
public createMagazineWithAmmo(magazineTpl: string, ammoTpl: string, magTemplate: ITemplateItem): Item[]
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
2023-11-16 22:42:06 +01:00
|
|
|
const magazine: Item[] = [{ _id: this.hashUtil.generate(), _tpl: magazineTpl }];
|
2023-10-10 13:03:20 +02:00
|
|
|
|
|
|
|
this.itemHelper.fillMagazineWithCartridge(magazine, magTemplate, ammoTpl, 1);
|
|
|
|
|
|
|
|
return magazine;
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add a specific number of cartridges to a bots inventory (defaults to vest and pockets)
|
|
|
|
* @param ammoTpl Ammo tpl to add to vest/pockets
|
|
|
|
* @param cartridgeCount number of cartridges to add to vest/pockets
|
|
|
|
* @param inventory bot inventory to add cartridges to
|
|
|
|
* @param equipmentSlotsToAddTo what equipment slots should bullets be added into
|
|
|
|
*/
|
2023-11-16 22:42:06 +01:00
|
|
|
public addAmmoIntoEquipmentSlots(
|
|
|
|
ammoTpl: string,
|
|
|
|
cartridgeCount: number,
|
|
|
|
inventory: Inventory,
|
|
|
|
equipmentSlotsToAddTo: EquipmentSlots[] = [EquipmentSlots.TACTICAL_VEST, EquipmentSlots.POCKETS],
|
|
|
|
): void
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
const ammoItems = this.itemHelper.splitStack({
|
|
|
|
_id: this.hashUtil.generate(),
|
|
|
|
_tpl: ammoTpl,
|
2023-11-16 22:42:06 +01:00
|
|
|
upd: { StackObjectsCount: cartridgeCount },
|
2023-03-03 16:23:46 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
for (const ammoItem of ammoItems)
|
|
|
|
{
|
2024-02-25 12:45:34 +01:00
|
|
|
const result = this.botGeneratorHelper.addItemWithChildrenToEquipmentSlot(
|
|
|
|
equipmentSlotsToAddTo,
|
|
|
|
ammoItem._id,
|
|
|
|
ammoItem._tpl,
|
|
|
|
[ammoItem],
|
|
|
|
inventory,
|
|
|
|
);
|
2023-10-10 13:03:20 +02:00
|
|
|
|
2023-11-24 16:35:19 +01:00
|
|
|
if (result !== ItemAddedResult.SUCCESS)
|
2023-10-10 13:03:20 +02:00
|
|
|
{
|
2023-11-24 17:05:58 +01:00
|
|
|
this.logger.debug(`Unable to add ammo: ${ammoItem._tpl} to bot inventory, ${ItemAddedResult[result]}`);
|
2023-11-24 16:35:19 +01:00
|
|
|
|
2023-11-25 11:04:17 +01:00
|
|
|
if (result === ItemAddedResult.NO_SPACE || result === ItemAddedResult.NO_CONTAINERS)
|
2023-11-24 16:35:19 +01:00
|
|
|
{
|
2023-12-08 17:27:34 +01:00
|
|
|
// If there's no space for 1 stack or no containers to hold item, there's no space for the others
|
2023-11-24 16:35:19 +01:00
|
|
|
break;
|
|
|
|
}
|
2023-10-10 13:03:20 +02:00
|
|
|
}
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a weapons default magazine template id
|
|
|
|
* @param weaponTemplate weapon to get default magazine for
|
|
|
|
* @returns tpl of magazine
|
|
|
|
*/
|
|
|
|
public getWeaponsDefaultMagazineTpl(weaponTemplate: ITemplateItem): string
|
|
|
|
{
|
|
|
|
return weaponTemplate._props.defMagType;
|
|
|
|
}
|
2023-11-16 22:42:06 +01:00
|
|
|
}
|