2023-03-03 16:23:46 +01:00
|
|
|
import { inject, injectable } from "tsyringe";
|
2024-05-21 19:59:04 +02:00
|
|
|
import { ItemHelper } from "@spt/helpers/ItemHelper";
|
|
|
|
import { ProfileHelper } from "@spt/helpers/ProfileHelper";
|
|
|
|
import { TraderHelper } from "@spt/helpers/TraderHelper";
|
|
|
|
import { Item } from "@spt/models/eft/common/tables/IItem";
|
|
|
|
import { ITemplateItem } from "@spt/models/eft/common/tables/ITemplateItem";
|
|
|
|
import { BaseClasses } from "@spt/models/enums/BaseClasses";
|
|
|
|
import { ConfigTypes } from "@spt/models/enums/ConfigTypes";
|
|
|
|
import { MessageType } from "@spt/models/enums/MessageType";
|
|
|
|
import { Traders } from "@spt/models/enums/Traders";
|
|
|
|
import { IQuestConfig } from "@spt/models/spt/config/IQuestConfig";
|
|
|
|
import { IRagfairConfig } from "@spt/models/spt/config/IRagfairConfig";
|
|
|
|
import { ILogger } from "@spt/models/spt/utils/ILogger";
|
|
|
|
import { ConfigServer } from "@spt/servers/ConfigServer";
|
|
|
|
import { DatabaseServer } from "@spt/servers/DatabaseServer";
|
|
|
|
import { SaveServer } from "@spt/servers/SaveServer";
|
|
|
|
import { ItemFilterService } from "@spt/services/ItemFilterService";
|
|
|
|
import { MailSendService } from "@spt/services/MailSendService";
|
|
|
|
import { ICloner } from "@spt/utils/cloners/ICloner";
|
|
|
|
import { RandomUtil } from "@spt/utils/RandomUtil";
|
|
|
|
import { TimeUtil } from "@spt/utils/TimeUtil";
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Helper class for common ragfair server actions
|
|
|
|
*/
|
|
|
|
@injectable()
|
|
|
|
export class RagfairServerHelper
|
|
|
|
{
|
|
|
|
protected ragfairConfig: IRagfairConfig;
|
|
|
|
protected questConfig: IQuestConfig;
|
2023-05-29 17:05:19 +02:00
|
|
|
protected static goodsReturnedTemplate = "5bdabfe486f7743e1665df6e 0"; // Your item was not sold
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
constructor(
|
2023-12-14 16:47:01 +01:00
|
|
|
@inject("WinstonLogger") protected logger: ILogger,
|
2023-03-03 16:23:46 +01:00
|
|
|
@inject("RandomUtil") protected randomUtil: RandomUtil,
|
2023-07-22 14:37:40 +02:00
|
|
|
@inject("TimeUtil") protected timeUtil: TimeUtil,
|
2023-03-03 16:23:46 +01:00
|
|
|
@inject("SaveServer") protected saveServer: SaveServer,
|
|
|
|
@inject("DatabaseServer") protected databaseServer: DatabaseServer,
|
|
|
|
@inject("ProfileHelper") protected profileHelper: ProfileHelper,
|
|
|
|
@inject("ItemHelper") protected itemHelper: ItemHelper,
|
2023-07-23 11:35:42 +02:00
|
|
|
@inject("TraderHelper") protected traderHelper: TraderHelper,
|
2023-07-22 14:13:01 +02:00
|
|
|
@inject("MailSendService") protected mailSendService: MailSendService,
|
2023-03-03 16:23:46 +01:00
|
|
|
@inject("ItemFilterService") protected itemFilterService: ItemFilterService,
|
2023-11-16 22:42:06 +01:00
|
|
|
@inject("ConfigServer") protected configServer: ConfigServer,
|
2024-05-13 19:58:17 +02:00
|
|
|
@inject("RecursiveCloner") protected cloner: ICloner,
|
2023-03-03 16:23:46 +01:00
|
|
|
)
|
|
|
|
{
|
|
|
|
this.ragfairConfig = this.configServer.getConfig(ConfigTypes.RAGFAIR);
|
|
|
|
this.questConfig = this.configServer.getConfig(ConfigTypes.QUEST);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Is item valid / on blacklist / quest item
|
2023-11-16 22:42:06 +01:00
|
|
|
* @param itemDetails
|
2023-03-03 16:23:46 +01:00
|
|
|
* @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
|
2023-07-22 14:13:01 +02:00
|
|
|
if (this.isItemOnCustomFleaBlacklist(itemDetails[1]._id))
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2024-03-04 20:36:31 +01:00
|
|
|
// Skip custom category blacklisted items
|
|
|
|
if (
|
|
|
|
blacklistConfig.enableCustomItemCategoryList
|
|
|
|
&& this.isItemCategoryOnCustomFleaBlacklist(itemDetails[1]._parent)
|
|
|
|
)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-03-03 16:23:46 +01:00
|
|
|
// Skip quest items
|
|
|
|
if (blacklistConfig.enableQuestList && this.itemHelper.isQuestItem(itemDetails[1]._id))
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-06-30 15:00:08 +02:00
|
|
|
// Don't include damaged ammo packs
|
2023-11-16 22:42:06 +01:00
|
|
|
if (
|
2024-05-17 21:32:41 +02:00
|
|
|
this.ragfairConfig.dynamic.blacklist.damagedAmmoPacks
|
|
|
|
&& itemDetails[1]._parent === BaseClasses.AMMO_BOX
|
2023-11-16 22:42:06 +01:00
|
|
|
&& itemDetails[1]._name.includes("_damaged")
|
|
|
|
)
|
2023-06-30 15:00:08 +02:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-03-03 16:23:46 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-07-22 14:13:01 +02:00
|
|
|
/**
|
|
|
|
* 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
|
|
|
|
*/
|
2023-07-22 14:14:40 +02:00
|
|
|
protected isItemOnCustomFleaBlacklist(itemTemplateId: string): boolean
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
if (!this.itemHelper.isValidItem(itemTemplateId))
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return this.ragfairConfig.dynamic.blacklist.custom.includes(itemTemplateId);
|
|
|
|
}
|
|
|
|
|
2024-03-04 20:36:31 +01:00
|
|
|
/**
|
|
|
|
* 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);
|
|
|
|
}
|
|
|
|
|
2023-07-22 14:13:01 +02:00
|
|
|
/**
|
|
|
|
* is supplied id a trader
|
2023-11-16 22:42:06 +01:00
|
|
|
* @param traderId
|
2023-07-22 14:13:01 +02:00
|
|
|
* @returns True if id was a trader
|
|
|
|
*/
|
|
|
|
public isTrader(traderId: string): boolean
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
2023-07-22 14:13:01 +02:00
|
|
|
return traderId in this.databaseServer.getTables().traders;
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
2023-07-22 14:13:01 +02:00
|
|
|
/**
|
|
|
|
* Is this user id the logged in player
|
|
|
|
* @param userId Id to test
|
|
|
|
* @returns True is the current player
|
|
|
|
*/
|
|
|
|
public isPlayer(userId: string): boolean
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
2023-07-22 14:13:01 +02:00
|
|
|
if (this.profileHelper.getPmcProfile(userId) !== undefined)
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-07-22 14:13:01 +02:00
|
|
|
/**
|
|
|
|
* 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
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
2023-07-22 14:13:01 +02:00
|
|
|
this.mailSendService.sendLocalisedNpcMessageToPlayer(
|
|
|
|
sessionID,
|
2023-07-23 11:35:42 +02:00
|
|
|
this.traderHelper.getTraderById(Traders.RAGMAN),
|
2023-07-22 14:13:01 +02:00
|
|
|
MessageType.MESSAGE_WITH_ITEMS,
|
2023-07-23 11:35:42 +02:00
|
|
|
RagfairServerHelper.goodsReturnedTemplate,
|
2023-07-22 14:13:01 +02:00
|
|
|
returnedItems,
|
2024-02-02 19:54:07 +01:00
|
|
|
this.timeUtil.getHoursAsSeconds(
|
|
|
|
this.databaseServer.getTables().globals.config.RagFair.yourOfferDidNotSellMaxStorageTimeInHour,
|
|
|
|
),
|
2023-07-22 14:13:01 +02:00
|
|
|
);
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
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
|
2023-11-16 22:42:06 +01:00
|
|
|
if (
|
|
|
|
isWeaponPreset
|
|
|
|
|| this.itemHelper.isOfBaseclasses(itemDetails[1]._id, this.ragfairConfig.dynamic.showAsSingleStack)
|
|
|
|
)
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
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));
|
|
|
|
}
|
|
|
|
|
2023-11-16 22:42:06 +01:00
|
|
|
const stackPercent = Math.round(
|
|
|
|
this.randomUtil.getInt(config.stackablePercent.min, config.stackablePercent.max),
|
|
|
|
);
|
2023-03-03 16:23:46 +01:00
|
|
|
|
2024-05-17 21:32:41 +02:00
|
|
|
return Math.round((maxStackCount / 100) * stackPercent);
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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)];
|
|
|
|
}
|
|
|
|
|
2023-12-14 16:47:01 +01:00
|
|
|
/**
|
|
|
|
* 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[]
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
2024-05-13 19:58:17 +02:00
|
|
|
const preset = this.cloner.clone(this.databaseServer.getTables().globals.ItemPresets[item._id]._items);
|
2024-01-14 22:12:56 +01:00
|
|
|
return this.itemHelper.reparentItemAndChildren(item, preset);
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
2023-12-14 16:47:01 +01:00
|
|
|
/**
|
|
|
|
* Possible bug, returns all items associated with an items tpl, could be multiple presets from globals.json
|
|
|
|
* @param item Preset item
|
2024-02-02 19:54:07 +01:00
|
|
|
* @returns
|
2023-12-14 16:47:01 +01:00
|
|
|
*/
|
2023-03-03 16:23:46 +01:00
|
|
|
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)
|
|
|
|
{
|
2024-05-13 19:58:17 +02:00
|
|
|
const presetItems = this.cloner.clone(
|
2023-11-16 22:42:06 +01:00
|
|
|
this.databaseServer.getTables().globals.ItemPresets[itemId]._items,
|
|
|
|
);
|
2024-01-14 22:12:56 +01:00
|
|
|
presets.push(this.itemHelper.reparentItemAndChildren(item, presetItems));
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return presets;
|
|
|
|
}
|
2023-11-16 22:42:06 +01:00
|
|
|
}
|