2023-03-03 16:23:46 +01:00
|
|
|
import { inject, injectable } from "tsyringe";
|
2024-05-21 19:59:04 +02:00
|
|
|
import { MinMax } from "@spt/models/common/MinMax";
|
|
|
|
import { Difficulty, IBotType } from "@spt/models/eft/common/tables/IBotType";
|
|
|
|
import { ConfigTypes } from "@spt/models/enums/ConfigTypes";
|
|
|
|
import { EquipmentFilters, IBotConfig, RandomisationDetails } from "@spt/models/spt/config/IBotConfig";
|
|
|
|
import { IPmcConfig } from "@spt/models/spt/config/IPmcConfig";
|
|
|
|
import { ILogger } from "@spt/models/spt/utils/ILogger";
|
|
|
|
import { ConfigServer } from "@spt/servers/ConfigServer";
|
|
|
|
import { DatabaseServer } from "@spt/servers/DatabaseServer";
|
|
|
|
import { RandomUtil } from "@spt/utils/RandomUtil";
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
@injectable()
|
|
|
|
export class BotHelper
|
|
|
|
{
|
|
|
|
protected botConfig: IBotConfig;
|
2023-10-10 13:03:20 +02:00
|
|
|
protected pmcConfig: IPmcConfig;
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
constructor(
|
|
|
|
@inject("WinstonLogger") protected logger: ILogger,
|
|
|
|
@inject("DatabaseServer") protected databaseServer: DatabaseServer,
|
|
|
|
@inject("RandomUtil") protected randomUtil: RandomUtil,
|
2023-11-13 17:07:59 +01:00
|
|
|
@inject("ConfigServer") protected configServer: ConfigServer,
|
2023-03-03 16:23:46 +01:00
|
|
|
)
|
|
|
|
{
|
|
|
|
this.botConfig = this.configServer.getConfig(ConfigTypes.BOT);
|
2023-10-10 13:03:20 +02:00
|
|
|
this.pmcConfig = this.configServer.getConfig(ConfigTypes.PMC);
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a template object for the specified botRole from bots.types db
|
|
|
|
* @param role botRole to get template for
|
|
|
|
* @returns IBotType object
|
|
|
|
*/
|
|
|
|
public getBotTemplate(role: string): IBotType
|
|
|
|
{
|
|
|
|
return this.databaseServer.getTables().bots.types[role.toLowerCase()];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Is the passed in bot role a PMC (usec/bear/pmc)
|
|
|
|
* @param botRole bot role to check
|
|
|
|
* @returns true if is pmc
|
|
|
|
*/
|
|
|
|
public isBotPmc(botRole: string): boolean
|
|
|
|
{
|
2024-05-08 05:57:08 +02:00
|
|
|
return ["usec", "bear", "pmc", "sptbear", "sptusec"].includes(botRole?.toLowerCase());
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public isBotBoss(botRole: string): boolean
|
|
|
|
{
|
2024-05-17 21:32:41 +02:00
|
|
|
return this.botConfig.bosses.some((x) => x.toLowerCase() === botRole?.toLowerCase());
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public isBotFollower(botRole: string): boolean
|
|
|
|
{
|
2023-11-05 21:29:10 +01:00
|
|
|
return botRole?.toLowerCase().startsWith("follower");
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add a bot to the FRIENDLY_BOT_TYPES array
|
|
|
|
* @param difficultySettings bot settings to alter
|
|
|
|
* @param typeToAdd bot type to add to friendly list
|
|
|
|
*/
|
|
|
|
public addBotToFriendlyList(difficultySettings: Difficulty, typeToAdd: string): void
|
|
|
|
{
|
|
|
|
const friendlyBotTypesKey = "FRIENDLY_BOT_TYPES";
|
|
|
|
|
|
|
|
// Null guard
|
|
|
|
if (!difficultySettings.Mind[friendlyBotTypesKey])
|
|
|
|
{
|
|
|
|
difficultySettings.Mind[friendlyBotTypesKey] = [];
|
|
|
|
}
|
|
|
|
|
|
|
|
(<string[]>difficultySettings.Mind[friendlyBotTypesKey]).push(typeToAdd);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add a bot to the REVENGE_BOT_TYPES array
|
|
|
|
* @param difficultySettings bot settings to alter
|
|
|
|
* @param typesToAdd bot type to add to revenge list
|
|
|
|
*/
|
|
|
|
public addBotToRevengeList(difficultySettings: Difficulty, typesToAdd: string[]): void
|
|
|
|
{
|
|
|
|
const revengePropKey = "REVENGE_BOT_TYPES";
|
|
|
|
|
|
|
|
// Nothing to add
|
|
|
|
if (!typesToAdd)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Null guard
|
|
|
|
if (!difficultySettings.Mind[revengePropKey])
|
|
|
|
{
|
|
|
|
difficultySettings.Mind[revengePropKey] = [];
|
|
|
|
}
|
|
|
|
|
|
|
|
const revengeArray = <string[]>difficultySettings.Mind[revengePropKey];
|
|
|
|
for (const botTypeToAdd of typesToAdd)
|
|
|
|
{
|
|
|
|
if (!revengeArray.includes(botTypeToAdd))
|
|
|
|
{
|
|
|
|
revengeArray.push(botTypeToAdd);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Choose if a bot should become a PMC by checking if bot type is allowed to become a Pmc in botConfig.convertFromChances and doing a random int check
|
|
|
|
* @param botRole the bot role to check if should be a pmc
|
|
|
|
* @returns true if should be a pmc
|
|
|
|
*/
|
|
|
|
public shouldBotBePmc(botRole: string): boolean
|
|
|
|
{
|
|
|
|
const botRoleLowered = botRole.toLowerCase();
|
|
|
|
|
|
|
|
// Handle when map waves have these types in the bot type
|
|
|
|
if (this.botRoleIsPmc(botRoleLowered))
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-10-10 13:03:20 +02:00
|
|
|
const botConvertMinMax = this.pmcConfig.convertIntoPmcChance[botRoleLowered];
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
// no bot type defined in config, default to false
|
|
|
|
if (!botConvertMinMax)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return this.rollChanceToBePmc(botRoleLowered, botConvertMinMax);
|
|
|
|
}
|
|
|
|
|
|
|
|
public rollChanceToBePmc(role: string, botConvertMinMax: MinMax): boolean
|
|
|
|
{
|
2024-05-17 21:32:41 +02:00
|
|
|
return (
|
|
|
|
role.toLowerCase() in this.pmcConfig.convertIntoPmcChance
|
|
|
|
&& this.randomUtil.getChance100(this.randomUtil.getInt(botConvertMinMax.min, botConvertMinMax.max))
|
|
|
|
);
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public botRoleIsPmc(botRole: string): boolean
|
|
|
|
{
|
2023-11-13 17:07:59 +01:00
|
|
|
return [this.pmcConfig.usecType.toLowerCase(), this.pmcConfig.bearType.toLowerCase()].includes(
|
|
|
|
botRole.toLowerCase(),
|
|
|
|
);
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get randomization settings for bot from config/bot.json
|
|
|
|
* @param botLevel level of bot
|
|
|
|
* @param botEquipConfig bot equipment json
|
|
|
|
* @returns RandomisationDetails
|
|
|
|
*/
|
|
|
|
public getBotRandomizationDetails(botLevel: number, botEquipConfig: EquipmentFilters): RandomisationDetails
|
|
|
|
{
|
|
|
|
// No randomisation details found, skip
|
|
|
|
if (!botEquipConfig || Object.keys(botEquipConfig).length === 0 || !botEquipConfig.randomisation)
|
|
|
|
{
|
|
|
|
return null;
|
|
|
|
}
|
2023-11-13 17:07:59 +01:00
|
|
|
|
2024-05-17 21:32:41 +02:00
|
|
|
return botEquipConfig.randomisation.find((x) => botLevel >= x.levelRange.min && botLevel <= x.levelRange.max);
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2023-10-10 13:03:20 +02:00
|
|
|
* Choose between sptBear and sptUsec at random based on the % defined in pmcConfig.isUsec
|
2023-03-03 16:23:46 +01:00
|
|
|
* @returns pmc role
|
|
|
|
*/
|
|
|
|
public getRandomizedPmcRole(): string
|
|
|
|
{
|
2024-05-17 21:32:41 +02:00
|
|
|
return this.randomUtil.getChance100(this.pmcConfig.isUsec) ? this.pmcConfig.usecType : this.pmcConfig.bearType;
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the corresponding side when sptBear or sptUsec is passed in
|
|
|
|
* @param botRole role to get side for
|
|
|
|
* @returns side (usec/bear)
|
|
|
|
*/
|
|
|
|
public getPmcSideByRole(botRole: string): string
|
|
|
|
{
|
|
|
|
switch (botRole.toLowerCase())
|
|
|
|
{
|
2023-10-10 13:03:20 +02:00
|
|
|
case this.pmcConfig.bearType.toLowerCase():
|
2023-03-03 16:23:46 +01:00
|
|
|
return "Bear";
|
2023-10-10 13:03:20 +02:00
|
|
|
case this.pmcConfig.usecType.toLowerCase():
|
2023-03-03 16:23:46 +01:00
|
|
|
return "Usec";
|
|
|
|
default:
|
|
|
|
return this.getRandomizedPmcSide();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a randomized PMC side based on bot config value 'isUsec'
|
|
|
|
* @returns pmc side as string
|
|
|
|
*/
|
|
|
|
protected getRandomizedPmcSide(): string
|
|
|
|
{
|
2024-05-08 05:57:08 +02:00
|
|
|
return this.randomUtil.getChance100(this.pmcConfig.isUsec) ? "Usec" : "Bear";
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
2024-05-22 14:26:35 +02:00
|
|
|
|
|
|
|
public getPmcNicknameOfMaxLength(userId: string, maxLength: number): string
|
|
|
|
{
|
|
|
|
// recurivse if name is longer than max characters allowed (15 characters)
|
|
|
|
const randomType = this.randomUtil.getInt(0, 1) === 0 ? "usec" : "bear";
|
|
|
|
const name
|
|
|
|
= this.randomUtil.getStringArrayValue(this.databaseServer.getTables().bots.types[randomType].firstName);
|
|
|
|
return name.length > maxLength
|
|
|
|
? this.getPmcNicknameOfMaxLength(userId, maxLength)
|
|
|
|
: name;
|
|
|
|
}
|
2023-11-13 17:07:59 +01:00
|
|
|
}
|