2023-03-08 14:26:32 +01:00
|
|
|
import { inject, injectable } from "tsyringe";
|
|
|
|
|
2023-10-19 19:21:17 +02:00
|
|
|
import { NotificationSendHelper } from "@spt-aki/helpers/NotificationSendHelper";
|
|
|
|
import { WeightedRandomHelper } from "@spt-aki/helpers/WeightedRandomHelper";
|
|
|
|
import { IPmcData } from "@spt-aki/models/eft/common/IPmcData";
|
|
|
|
import { Aggressor, Victim } from "@spt-aki/models/eft/common/tables/IBotBase";
|
|
|
|
import { IUserDialogInfo } from "@spt-aki/models/eft/profile/IAkiProfile";
|
|
|
|
import { ConfigTypes } from "@spt-aki/models/enums/ConfigTypes";
|
|
|
|
import { MemberCategory } from "@spt-aki/models/enums/MemberCategory";
|
|
|
|
import { MessageType } from "@spt-aki/models/enums/MessageType";
|
|
|
|
import { IPmcChatResponse } from "@spt-aki/models/spt/config/IPmChatResponse";
|
|
|
|
import { ILogger } from "@spt-aki/models/spt/utils/ILogger";
|
|
|
|
import { ConfigServer } from "@spt-aki/servers/ConfigServer";
|
|
|
|
import { LocalisationService } from "@spt-aki/services/LocalisationService";
|
|
|
|
import { MatchBotDetailsCacheService } from "@spt-aki/services/MatchBotDetailsCacheService";
|
2024-01-06 14:49:48 +01:00
|
|
|
import { HashUtil } from "@spt-aki/utils/HashUtil";
|
2023-10-19 19:21:17 +02:00
|
|
|
import { RandomUtil } from "@spt-aki/utils/RandomUtil";
|
2023-03-08 14:26:32 +01:00
|
|
|
|
|
|
|
@injectable()
|
|
|
|
export class PmcChatResponseService
|
|
|
|
{
|
|
|
|
protected pmcResponsesConfig: IPmcChatResponse;
|
|
|
|
|
|
|
|
constructor(
|
2023-03-22 11:25:34 +01:00
|
|
|
@inject("WinstonLogger") protected logger: ILogger,
|
2024-01-06 14:49:48 +01:00
|
|
|
@inject("HashUtil") protected hashUtil: HashUtil,
|
2023-03-08 14:26:32 +01:00
|
|
|
@inject("RandomUtil") protected randomUtil: RandomUtil,
|
|
|
|
@inject("NotificationSendHelper") protected notificationSendHelper: NotificationSendHelper,
|
2023-03-22 11:25:34 +01:00
|
|
|
@inject("MatchBotDetailsCacheService") protected matchBotDetailsCacheService: MatchBotDetailsCacheService,
|
2023-03-08 14:26:32 +01:00
|
|
|
@inject("LocalisationService") protected localisationService: LocalisationService,
|
|
|
|
@inject("WeightedRandomHelper") protected weightedRandomHelper: WeightedRandomHelper,
|
2023-11-16 02:35:05 +01:00
|
|
|
@inject("ConfigServer") protected configServer: ConfigServer,
|
2023-03-08 14:26:32 +01:00
|
|
|
)
|
|
|
|
{
|
|
|
|
this.pmcResponsesConfig = this.configServer.getConfig(ConfigTypes.PMC_CHAT_RESPONSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2023-03-13 16:23:34 +01:00
|
|
|
* For each PMC victim of the player, have a chance to send a message to the player, can be positive or negative
|
2023-03-08 14:26:32 +01:00
|
|
|
* @param sessionId Session id
|
|
|
|
* @param pmcVictims Array of bots killed by player
|
2023-03-22 18:12:19 +01:00
|
|
|
* @param pmcData Player profile
|
2023-03-08 14:26:32 +01:00
|
|
|
*/
|
2023-03-22 18:12:19 +01:00
|
|
|
public sendVictimResponse(sessionId: string, pmcVictims: Victim[], pmcData: IPmcData): void
|
2023-03-08 14:26:32 +01:00
|
|
|
{
|
2023-03-13 16:23:34 +01:00
|
|
|
for (const victim of pmcVictims)
|
|
|
|
{
|
|
|
|
if (!this.randomUtil.getChance100(this.pmcResponsesConfig.victim.responseChancePercent))
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
const victimDetails = this.getVictimDetails(victim);
|
2023-03-22 18:12:19 +01:00
|
|
|
const message = this.chooseMessage(true, pmcData);
|
2023-11-16 02:35:05 +01:00
|
|
|
this.notificationSendHelper.sendMessageToPlayer(
|
|
|
|
sessionId,
|
|
|
|
victimDetails,
|
|
|
|
message,
|
|
|
|
MessageType.USER_MESSAGE,
|
|
|
|
);
|
|
|
|
}
|
2023-03-08 14:26:32 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Not fully implemented yet, needs method of acquiring killers details after raid
|
|
|
|
* @param sessionId Session id
|
|
|
|
* @param pmcData Players profile
|
2023-03-22 18:12:19 +01:00
|
|
|
* @param killer The bot who killed the player
|
2023-03-08 14:26:32 +01:00
|
|
|
*/
|
2023-03-22 11:25:34 +01:00
|
|
|
public sendKillerResponse(sessionId: string, pmcData: IPmcData, killer: Aggressor): void
|
2023-03-08 14:26:32 +01:00
|
|
|
{
|
2023-03-23 21:18:53 +01:00
|
|
|
if (!killer)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-03-22 11:25:34 +01:00
|
|
|
if (!this.randomUtil.getChance100(this.pmcResponsesConfig.killer.responseChancePercent))
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// find bot by name in cache
|
2023-11-16 02:35:05 +01:00
|
|
|
const killerDetailsInCache = this.matchBotDetailsCacheService.getBotByNameAndSide(
|
|
|
|
killer.Name.trim(),
|
|
|
|
killer.Side,
|
|
|
|
);
|
2023-03-22 11:25:34 +01:00
|
|
|
if (!killerDetailsInCache)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If kill was not a PMC, skip
|
|
|
|
if (!["sptUsec", "sptBear"].includes(killerDetailsInCache.Info.Settings.Role))
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const killerDetails: IUserDialogInfo = {
|
|
|
|
_id: killerDetailsInCache._id,
|
2024-01-06 14:49:48 +01:00
|
|
|
aid: this.hashUtil.generateAccountId(), // TODO- pass correct value
|
2024-01-06 14:40:00 +01:00
|
|
|
Info: {
|
2023-03-22 11:25:34 +01:00
|
|
|
Nickname: killerDetailsInCache.Info.Nickname,
|
|
|
|
Side: killerDetailsInCache.Info.Side,
|
|
|
|
Level: killerDetailsInCache.Info.Level,
|
2023-11-16 02:35:05 +01:00
|
|
|
MemberCategory: killerDetailsInCache.Info.MemberCategory,
|
|
|
|
},
|
2023-03-08 14:26:32 +01:00
|
|
|
};
|
|
|
|
|
2023-03-22 18:12:19 +01:00
|
|
|
const message = this.chooseMessage(false, pmcData);
|
2023-03-22 11:25:34 +01:00
|
|
|
if (!message)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2023-03-08 14:26:32 +01:00
|
|
|
|
2023-03-22 11:25:34 +01:00
|
|
|
this.notificationSendHelper.sendMessageToPlayer(sessionId, killerDetails, message, MessageType.USER_MESSAGE);
|
2023-03-08 14:26:32 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Choose a localised message to send the player (different if sender was killed or killed player)
|
2023-03-22 18:12:19 +01:00
|
|
|
* @param isVictim Is the message coming from a bot killed by the player
|
|
|
|
* @param pmcData Player profile
|
|
|
|
* @returns Message from PMC to player
|
2023-03-08 14:26:32 +01:00
|
|
|
*/
|
2023-03-22 18:12:19 +01:00
|
|
|
protected chooseMessage(isVictim: boolean, pmcData: IPmcData): string
|
2023-03-08 14:26:32 +01:00
|
|
|
{
|
|
|
|
// Positive/negative etc
|
|
|
|
const responseType = this.chooseResponseType(isVictim);
|
|
|
|
|
|
|
|
// Get all locale keys
|
|
|
|
const possibleResponseLocaleKeys = this.getResponseLocaleKeys(responseType, isVictim);
|
2023-03-22 11:25:34 +01:00
|
|
|
if (possibleResponseLocaleKeys.length === 0)
|
|
|
|
{
|
2023-07-19 14:16:45 +02:00
|
|
|
this.logger.warning(this.localisationService.getText("pmcresponse-unable_to_find_key", responseType));
|
2023-03-22 11:25:34 +01:00
|
|
|
|
|
|
|
return undefined;
|
|
|
|
}
|
2023-03-08 14:26:32 +01:00
|
|
|
|
|
|
|
// Choose random response from above list and request it from localisation service
|
2023-11-16 02:35:05 +01:00
|
|
|
let responseText = this.localisationService.getText(this.randomUtil.getArrayValue(possibleResponseLocaleKeys), {
|
|
|
|
playerName: pmcData.Info.Nickname,
|
|
|
|
playerLevel: pmcData.Info.Level,
|
|
|
|
playerSide: pmcData.Info.Side,
|
|
|
|
});
|
2023-03-13 15:02:39 +01:00
|
|
|
|
|
|
|
if (this.appendSuffixToMessageEnd(isVictim))
|
2023-03-08 14:26:32 +01:00
|
|
|
{
|
2023-11-16 02:35:05 +01:00
|
|
|
const suffixText = this.localisationService.getText(
|
|
|
|
this.randomUtil.getArrayValue(this.getResponseSuffixLocaleKeys()),
|
|
|
|
);
|
2023-03-13 15:02:39 +01:00
|
|
|
responseText += ` ${suffixText}`;
|
2023-03-08 14:26:32 +01:00
|
|
|
}
|
2023-11-16 02:35:05 +01:00
|
|
|
|
2023-03-08 14:26:32 +01:00
|
|
|
if (this.stripCapitalistion(isVictim))
|
|
|
|
{
|
|
|
|
responseText = responseText.toLowerCase();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.allCaps(isVictim))
|
|
|
|
{
|
|
|
|
responseText = responseText.toUpperCase();
|
|
|
|
}
|
|
|
|
|
|
|
|
return responseText;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Should capitalisation be stripped from the message response before sending
|
|
|
|
* @param isVictim Was responder a victim of player
|
|
|
|
* @returns true = should be stripped
|
|
|
|
*/
|
|
|
|
protected stripCapitalistion(isVictim: boolean): boolean
|
|
|
|
{
|
|
|
|
const chance = isVictim
|
|
|
|
? this.pmcResponsesConfig.victim.stripCapitalisationChancePercent
|
|
|
|
: this.pmcResponsesConfig.killer.stripCapitalisationChancePercent;
|
|
|
|
|
|
|
|
return this.randomUtil.getChance100(chance);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Should capitalisation be stripped from the message response before sending
|
|
|
|
* @param isVictim Was responder a victim of player
|
|
|
|
* @returns true = should be stripped
|
|
|
|
*/
|
|
|
|
protected allCaps(isVictim: boolean): boolean
|
|
|
|
{
|
|
|
|
const chance = isVictim
|
|
|
|
? this.pmcResponsesConfig.victim.allCapsChancePercent
|
|
|
|
: this.pmcResponsesConfig.killer.allCapsChancePercent;
|
|
|
|
|
|
|
|
return this.randomUtil.getChance100(chance);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2023-03-13 15:02:39 +01:00
|
|
|
* Should a suffix be appended to the end of the message being sent to player
|
2023-03-08 14:26:32 +01:00
|
|
|
* @param isVictim Was responder a victim of player
|
|
|
|
* @returns true = should be stripped
|
|
|
|
*/
|
2023-03-13 15:02:39 +01:00
|
|
|
appendSuffixToMessageEnd(isVictim: boolean): boolean
|
2023-03-08 14:26:32 +01:00
|
|
|
{
|
|
|
|
const chance = isVictim
|
|
|
|
? this.pmcResponsesConfig.victim.appendBroToMessageEndChancePercent
|
|
|
|
: this.pmcResponsesConfig.killer.appendBroToMessageEndChancePercent;
|
|
|
|
|
|
|
|
return this.randomUtil.getChance100(chance);
|
|
|
|
}
|
2023-11-16 02:35:05 +01:00
|
|
|
|
2023-03-08 14:26:32 +01:00
|
|
|
/**
|
|
|
|
* Choose a type of response based on the weightings in pmc response config
|
|
|
|
* @param isVictim Was responder killed by player
|
|
|
|
* @returns Response type (positive/negative)
|
|
|
|
*/
|
|
|
|
protected chooseResponseType(isVictim = true): string
|
|
|
|
{
|
|
|
|
const responseWeights = isVictim
|
|
|
|
? this.pmcResponsesConfig.victim.responseTypeWeights
|
|
|
|
: this.pmcResponsesConfig.killer.responseTypeWeights;
|
|
|
|
|
2023-10-10 13:03:20 +02:00
|
|
|
return this.weightedRandomHelper.getWeightedValue<string>(responseWeights);
|
2023-03-08 14:26:32 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get locale keys related to the type of response to send (victim/killer)
|
|
|
|
* @param keyType Positive/negative
|
|
|
|
* @param isVictim Was responder killed by player
|
2023-11-16 02:35:05 +01:00
|
|
|
* @returns
|
2023-03-08 14:26:32 +01:00
|
|
|
*/
|
|
|
|
protected getResponseLocaleKeys(keyType: string, isVictim = true): string[]
|
|
|
|
{
|
|
|
|
const keyBase = isVictim ? "pmcresponse-victim_" : "pmcresponse-killer_";
|
|
|
|
const keys = this.localisationService.getKeys();
|
|
|
|
|
2023-11-16 02:35:05 +01:00
|
|
|
return keys.filter((x) => x.startsWith(`${keyBase}${keyType}`));
|
2023-03-08 14:26:32 +01:00
|
|
|
}
|
|
|
|
|
2023-03-13 15:02:39 +01:00
|
|
|
/**
|
|
|
|
* Get all locale keys that start with `pmcresponse-suffix`
|
|
|
|
* @returns array of keys
|
|
|
|
*/
|
|
|
|
protected getResponseSuffixLocaleKeys(): string[]
|
|
|
|
{
|
|
|
|
const keys = this.localisationService.getKeys();
|
|
|
|
|
2023-11-16 02:35:05 +01:00
|
|
|
return keys.filter((x) => x.startsWith("pmcresponse-suffix"));
|
2023-03-13 15:02:39 +01:00
|
|
|
}
|
|
|
|
|
2023-03-08 14:26:32 +01:00
|
|
|
/**
|
|
|
|
* Randomly draw a victim of the the array and return thier details
|
|
|
|
* @param pmcVictims Possible victims to choose from
|
|
|
|
* @returns IUserDialogInfo
|
|
|
|
*/
|
|
|
|
protected chooseRandomVictim(pmcVictims: Victim[]): IUserDialogInfo
|
|
|
|
{
|
|
|
|
const randomVictim = this.randomUtil.getArrayValue(pmcVictims);
|
|
|
|
|
2023-03-13 16:23:34 +01:00
|
|
|
return this.getVictimDetails(randomVictim);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Convert a victim object into a IUserDialogInfo object
|
|
|
|
* @param pmcVictim victim to convert
|
|
|
|
* @returns IUserDialogInfo
|
|
|
|
*/
|
|
|
|
protected getVictimDetails(pmcVictim: Victim): IUserDialogInfo
|
|
|
|
{
|
2023-11-16 02:35:05 +01:00
|
|
|
const categories = [
|
|
|
|
MemberCategory.UNIQUE_ID,
|
|
|
|
MemberCategory.DEFAULT,
|
|
|
|
MemberCategory.DEFAULT,
|
|
|
|
MemberCategory.DEFAULT,
|
|
|
|
MemberCategory.DEFAULT,
|
|
|
|
MemberCategory.DEFAULT,
|
|
|
|
MemberCategory.DEFAULT,
|
|
|
|
MemberCategory.SHERPA,
|
|
|
|
MemberCategory.DEVELOPER,
|
|
|
|
];
|
|
|
|
return {
|
|
|
|
_id: pmcVictim.Name,
|
2024-01-06 14:49:48 +01:00
|
|
|
aid: this.hashUtil.generateAccountId(), // TODO- pass correct value
|
2024-01-06 14:40:00 +01:00
|
|
|
Info: {
|
2023-11-16 02:35:05 +01:00
|
|
|
Nickname: pmcVictim.Name,
|
|
|
|
Level: pmcVictim.Level,
|
|
|
|
Side: pmcVictim.Side,
|
|
|
|
MemberCategory: this.randomUtil.getArrayValue(categories),
|
|
|
|
},
|
|
|
|
};
|
2023-03-08 14:26:32 +01:00
|
|
|
}
|
2023-11-16 02:35:05 +01:00
|
|
|
}
|