import { inject, injectable } from "tsyringe"; import { DialogueHelper } from "@spt-aki/helpers/DialogueHelper"; import { ItemHelper } from "@spt-aki/helpers/ItemHelper"; import { ProfileHelper } from "@spt-aki/helpers/ProfileHelper"; import { TraderHelper } from "@spt-aki/helpers/TraderHelper"; import { Item } from "@spt-aki/models/eft/common/tables/IItem"; import { ITemplateItem } 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 { MemberCategory } from "@spt-aki/models/enums/MemberCategory"; import { MessageType } from "@spt-aki/models/enums/MessageType"; import { Traders } from "@spt-aki/models/enums/Traders"; import { IQuestConfig } from "@spt-aki/models/spt/config/IQuestConfig"; import { IRagfairConfig } from "@spt-aki/models/spt/config/IRagfairConfig"; import { ILogger } from "@spt-aki/models/spt/utils/ILogger"; import { ConfigServer } from "@spt-aki/servers/ConfigServer"; import { DatabaseServer } from "@spt-aki/servers/DatabaseServer"; import { SaveServer } from "@spt-aki/servers/SaveServer"; import { ItemFilterService } from "@spt-aki/services/ItemFilterService"; import { LocaleService } from "@spt-aki/services/LocaleService"; import { MailSendService } from "@spt-aki/services/MailSendService"; import { HashUtil } from "@spt-aki/utils/HashUtil"; import { JsonUtil } from "@spt-aki/utils/JsonUtil"; import { RandomUtil } from "@spt-aki/utils/RandomUtil"; import { TimeUtil } from "@spt-aki/utils/TimeUtil"; /** * Helper class for common ragfair server actions */ @injectable() export class RagfairServerHelper { protected ragfairConfig: IRagfairConfig; protected questConfig: IQuestConfig; protected static goodsReturnedTemplate = "5bdabfe486f7743e1665df6e 0"; // Your item was not sold constructor( @inject("WinstonLogger") protected logger: ILogger, @inject("RandomUtil") protected randomUtil: RandomUtil, @inject("HashUtil") protected hashUtil: HashUtil, @inject("TimeUtil") protected timeUtil: TimeUtil, @inject("SaveServer") protected saveServer: SaveServer, @inject("DatabaseServer") protected databaseServer: DatabaseServer, @inject("ProfileHelper") protected profileHelper: ProfileHelper, @inject("ItemHelper") protected itemHelper: ItemHelper, @inject("LocaleService") protected localeService: LocaleService, @inject("DialogueHelper") protected dialogueHelper: DialogueHelper, @inject("TraderHelper") protected traderHelper: TraderHelper, @inject("JsonUtil") protected jsonUtil: JsonUtil, @inject("MailSendService") protected mailSendService: MailSendService, @inject("ItemFilterService") protected itemFilterService: ItemFilterService, @inject("ConfigServer") protected configServer: ConfigServer, ) { this.ragfairConfig = this.configServer.getConfig(ConfigTypes.RAGFAIR); this.questConfig = this.configServer.getConfig(ConfigTypes.QUEST); } /** * Is item valid / on blacklist / quest item * @param itemDetails * @returns boolean */ public isItemValidRagfairItem(itemDetails: [boolean, ITemplateItem]): boolean { const blacklistConfig = this.ragfairConfig.dynamic.blacklist; // Skip invalid items if (!itemDetails[0]) { return false; } // Skip blacklisted items if (this.itemFilterService.isItemBlacklisted(itemDetails[1]._id)) { return false; } // Skip bsg blacklisted items if (blacklistConfig.enableBsgList && !itemDetails[1]._props.CanSellOnRagfair) { return false; } // Skip custom blacklisted items if (this.isItemOnCustomFleaBlacklist(itemDetails[1]._id)) { return false; } // Skip custom category blacklisted items if ( blacklistConfig.enableCustomItemCategoryList && this.isItemCategoryOnCustomFleaBlacklist(itemDetails[1]._parent) ) { return false; } // Skip quest items if (blacklistConfig.enableQuestList && this.itemHelper.isQuestItem(itemDetails[1]._id)) { return false; } // Don't include damaged ammo packs if ( this.ragfairConfig.dynamic.blacklist.damagedAmmoPacks && itemDetails[1]._parent === BaseClasses.AMMO_BOX && itemDetails[1]._name.includes("_damaged") ) { return false; } return true; } /** * Is supplied item tpl on the ragfair custom blacklist from configs/ragfair.json/dynamic * @param itemTemplateId Item tpl to check is blacklisted * @returns True if its blacklsited */ protected isItemOnCustomFleaBlacklist(itemTemplateId: string): boolean { if (!this.itemHelper.isValidItem(itemTemplateId)) { return true; } return this.ragfairConfig.dynamic.blacklist.custom.includes(itemTemplateId); } /** * Is supplied parent id on the ragfair custom item category blacklist * @param parentId Parent Id to check is blacklisted * @returns true if blacklisted */ protected isItemCategoryOnCustomFleaBlacklist(itemParentId: string): boolean { return this.ragfairConfig.dynamic.blacklist.customItemCategoryList.includes(itemParentId); } /** * is supplied id a trader * @param traderId * @returns True if id was a trader */ public isTrader(traderId: string): boolean { return traderId in this.databaseServer.getTables().traders; } /** * Is this user id the logged in player * @param userId Id to test * @returns True is the current player */ public isPlayer(userId: string): boolean { if (this.profileHelper.getPmcProfile(userId) !== undefined) { return true; } return false; } /** * Send items back to player * @param sessionID Player to send items to * @param returnedItems Items to send to player */ public returnItems(sessionID: string, returnedItems: Item[]): void { this.mailSendService.sendLocalisedNpcMessageToPlayer( sessionID, this.traderHelper.getTraderById(Traders.RAGMAN), MessageType.MESSAGE_WITH_ITEMS, RagfairServerHelper.goodsReturnedTemplate, returnedItems, this.timeUtil.getHoursAsSeconds( this.databaseServer.getTables().globals.config.RagFair.yourOfferDidNotSellMaxStorageTimeInHour, ), ); } public calculateDynamicStackCount(tplId: string, isWeaponPreset: boolean): number { const config = this.ragfairConfig.dynamic; // Lookup item details - check if item not found const itemDetails = this.itemHelper.getItem(tplId); if (!itemDetails[0]) { throw new Error(`Item with tpl ${tplId} not found. Unable to generate a dynamic stack count.`); } // Item Types to return one of if ( isWeaponPreset || this.itemHelper.isOfBaseclasses(itemDetails[1]._id, this.ragfairConfig.dynamic.showAsSingleStack) ) { return 1; } // Get max stack count const maxStackCount = itemDetails[1]._props.StackMaxSize; // non-stackable - use different values to calculate stack size if (!maxStackCount || maxStackCount === 1) { return Math.round(this.randomUtil.getInt(config.nonStackableCount.min, config.nonStackableCount.max)); } const stackPercent = Math.round( this.randomUtil.getInt(config.stackablePercent.min, config.stackablePercent.max), ); return Math.round(maxStackCount / 100 * stackPercent); } /** * Choose a currency at random with bias * @returns currency tpl */ public getDynamicOfferCurrency(): string { const currencies = this.ragfairConfig.dynamic.currencies; const bias: string[] = []; for (const item in currencies) { for (let i = 0; i < currencies[item]; i++) { bias.push(item); } } return bias[Math.floor(Math.random() * bias.length)]; } public getMemberType(userID: string): MemberCategory { if (this.isPlayer(userID)) { // player offer return this.saveServer.getProfile(userID).characters.pmc.Info.AccountType; } if (this.isTrader(userID)) { // trader offer return MemberCategory.TRADER; } // generated offer return MemberCategory.DEFAULT; } /** * Get a player or traders nickname from their profile by their user id * @param userID Sessionid/userid * @returns Nickname of individual */ public getNickname(userID: string): string { if (this.isPlayer(userID)) { // player offer return this.saveServer.getProfile(userID).characters.pmc.Info.Nickname; } if (this.isTrader(userID)) { // trader offer return this.databaseServer.getTables().traders[userID].base.nickname; } // generated offer // recurivse if name is longer than max characters allowed (15 characters) const type = this.randomUtil.getInt(0, 1) === 0 ? "usec" : "bear"; const name = this.randomUtil.getStringArrayValue(this.databaseServer.getTables().bots.types[type].firstName); return name.length > 15 ? this.getNickname(userID) : name; } /** * Given a preset id from globals.json, return an array of items[] with unique ids * @param item Preset item * @returns Array of weapon and its children */ public getPresetItems(item: Item): Item[] { const preset = this.jsonUtil.clone(this.databaseServer.getTables().globals.ItemPresets[item._id]._items); return this.itemHelper.reparentItemAndChildren(item, preset); } /** * Possible bug, returns all items associated with an items tpl, could be multiple presets from globals.json * @param item Preset item * @returns */ public getPresetItemsByTpl(item: Item): Item[] { const presets = []; for (const itemId in this.databaseServer.getTables().globals.ItemPresets) { if (this.databaseServer.getTables().globals.ItemPresets[itemId]._items[0]._tpl === item._tpl) { const presetItems = this.jsonUtil.clone( this.databaseServer.getTables().globals.ItemPresets[itemId]._items, ); presets.push(this.itemHelper.reparentItemAndChildren(item, presetItems)); } } return presets; } }