2023-03-03 16:23:46 +01:00
|
|
|
import { inject, injectable } from "tsyringe";
|
|
|
|
|
2023-10-19 19:21:17 +02:00
|
|
|
import { RagfairAssortGenerator } from "@spt-aki/generators/RagfairAssortGenerator";
|
|
|
|
import { RagfairOfferGenerator } from "@spt-aki/generators/RagfairOfferGenerator";
|
|
|
|
import { AssortHelper } from "@spt-aki/helpers/AssortHelper";
|
|
|
|
import { PaymentHelper } from "@spt-aki/helpers/PaymentHelper";
|
|
|
|
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 { ITrader, ITraderAssort } from "@spt-aki/models/eft/common/tables/ITrader";
|
|
|
|
import { ConfigTypes } from "@spt-aki/models/enums/ConfigTypes";
|
|
|
|
import { Traders } from "@spt-aki/models/enums/Traders";
|
|
|
|
import { ITraderConfig } from "@spt-aki/models/spt/config/ITraderConfig";
|
|
|
|
import { ILogger } from "@spt-aki/models/spt/utils/ILogger";
|
|
|
|
import { ConfigServer } from "@spt-aki/servers/ConfigServer";
|
|
|
|
import { DatabaseServer } from "@spt-aki/servers/DatabaseServer";
|
|
|
|
import { FenceService } from "@spt-aki/services/FenceService";
|
|
|
|
import { LocalisationService } from "@spt-aki/services/LocalisationService";
|
|
|
|
import { TraderAssortService } from "@spt-aki/services/TraderAssortService";
|
|
|
|
import { TraderPurchasePersisterService } from "@spt-aki/services/TraderPurchasePersisterService";
|
|
|
|
import { JsonUtil } from "@spt-aki/utils/JsonUtil";
|
|
|
|
import { MathUtil } from "@spt-aki/utils/MathUtil";
|
|
|
|
import { TimeUtil } from "@spt-aki/utils/TimeUtil";
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
@injectable()
|
|
|
|
export class TraderAssortHelper
|
|
|
|
{
|
|
|
|
protected traderConfig: ITraderConfig;
|
2023-11-16 02:35:05 +01:00
|
|
|
protected mergedQuestAssorts: Record<string, Record<string, string>> = { started: {}, success: {}, fail: {} };
|
2023-03-03 16:23:46 +01:00
|
|
|
protected createdMergedQuestAssorts = false;
|
|
|
|
|
|
|
|
constructor(
|
|
|
|
@inject("WinstonLogger") protected logger: ILogger,
|
|
|
|
@inject("JsonUtil") protected jsonUtil: JsonUtil,
|
|
|
|
@inject("MathUtil") protected mathUtil: MathUtil,
|
|
|
|
@inject("TimeUtil") protected timeUtil: TimeUtil,
|
|
|
|
@inject("DatabaseServer") protected databaseServer: DatabaseServer,
|
|
|
|
@inject("ProfileHelper") protected profileHelper: ProfileHelper,
|
|
|
|
@inject("AssortHelper") protected assortHelper: AssortHelper,
|
|
|
|
@inject("PaymentHelper") protected paymentHelper: PaymentHelper,
|
|
|
|
@inject("RagfairAssortGenerator") protected ragfairAssortGenerator: RagfairAssortGenerator,
|
|
|
|
@inject("RagfairOfferGenerator") protected ragfairOfferGenerator: RagfairOfferGenerator,
|
|
|
|
@inject("TraderAssortService") protected traderAssortService: TraderAssortService,
|
|
|
|
@inject("LocalisationService") protected localisationService: LocalisationService,
|
2023-11-16 02:35:05 +01:00
|
|
|
@inject("TraderPurchasePersisterService") protected traderPurchasePersisterService:
|
|
|
|
TraderPurchasePersisterService,
|
2023-03-03 16:23:46 +01:00
|
|
|
@inject("TraderHelper") protected traderHelper: TraderHelper,
|
|
|
|
@inject("FenceService") protected fenceService: FenceService,
|
2023-11-16 02:35:05 +01:00
|
|
|
@inject("ConfigServer") protected configServer: ConfigServer,
|
2023-03-03 16:23:46 +01:00
|
|
|
)
|
|
|
|
{
|
|
|
|
this.traderConfig = this.configServer.getConfig(ConfigTypes.TRADER);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a traders assorts
|
|
|
|
* Can be used for returning ragfair / fence assorts
|
|
|
|
* Filter out assorts not unlocked due to level OR quest completion
|
|
|
|
* @param sessionId session id
|
|
|
|
* @param traderId traders id
|
2023-12-05 21:41:43 +01:00
|
|
|
* @param flea Should assorts player hasn't unlocked be returned - default false
|
2023-03-03 16:23:46 +01:00
|
|
|
* @returns a traders' assorts
|
|
|
|
*/
|
|
|
|
public getAssort(sessionId: string, traderId: string, flea = false): ITraderAssort
|
|
|
|
{
|
|
|
|
// Special case for getting ragfair items as they're dynamically generated
|
|
|
|
if (traderId === "ragfair")
|
|
|
|
{
|
|
|
|
return this.getRagfairDataAsTraderAssort();
|
|
|
|
}
|
|
|
|
|
2024-02-05 15:43:46 +01:00
|
|
|
const traderClone = this.jsonUtil.clone(this.databaseServer.getTables().traders[traderId]);
|
2023-03-03 16:23:46 +01:00
|
|
|
const pmcProfile = this.profileHelper.getPmcProfile(sessionId);
|
|
|
|
|
|
|
|
if (traderId === Traders.FENCE)
|
|
|
|
{
|
|
|
|
return this.fenceService.getFenceAssorts(pmcProfile);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Strip assorts player should not see yet
|
|
|
|
if (!flea)
|
2023-11-16 02:35:05 +01:00
|
|
|
{
|
2024-02-05 15:43:46 +01:00
|
|
|
traderClone.assort = this.assortHelper.stripLockedLoyaltyAssort(pmcProfile, traderId, traderClone.assort);
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
2024-02-05 15:43:46 +01:00
|
|
|
this.resetBuyRestrictionCurrentValue(traderClone.assort.items);
|
2024-01-20 01:19:13 +01:00
|
|
|
|
2023-03-03 16:23:46 +01:00
|
|
|
// Append nextResupply value to assorts so client knows when refresh is occuring
|
2024-02-05 15:43:46 +01:00
|
|
|
traderClone.assort.nextResupply = traderClone.base.nextResupply;
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
// Adjust displayed assort counts based on values stored in profile
|
2023-11-16 02:35:05 +01:00
|
|
|
const assortPurchasesfromTrader = this.traderPurchasePersisterService.getProfileTraderPurchases(
|
|
|
|
sessionId,
|
|
|
|
traderId,
|
|
|
|
);
|
2023-03-03 16:23:46 +01:00
|
|
|
for (const assortId in assortPurchasesfromTrader)
|
|
|
|
{
|
|
|
|
// Find assort we want to update current buy count of
|
2024-02-05 15:43:46 +01:00
|
|
|
const assortToAdjust = traderClone.assort.items.find((x) => x._id === assortId);
|
2023-03-03 16:23:46 +01:00
|
|
|
if (!assortToAdjust)
|
|
|
|
{
|
2023-11-16 02:35:05 +01:00
|
|
|
this.logger.debug(
|
2024-02-05 15:43:46 +01:00
|
|
|
`Cannot find trader: ${traderClone.base.nickname} assort: ${assortId} to adjust BuyRestrictionCurrent value, skipping`,
|
2023-11-16 02:35:05 +01:00
|
|
|
);
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!assortToAdjust.upd)
|
|
|
|
{
|
2023-11-16 02:35:05 +01:00
|
|
|
this.logger.debug(
|
|
|
|
`Unable to adjust assort ${assortToAdjust._id} item: ${assortToAdjust._tpl} BuyRestrictionCurrent value, assort has an undefined upd object`,
|
|
|
|
);
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
assortToAdjust.upd.BuyRestrictionCurrent = assortPurchasesfromTrader[assortId].count;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get rid of quest locked assorts
|
|
|
|
if (!this.createdMergedQuestAssorts)
|
|
|
|
{
|
|
|
|
this.hydrateMergedQuestAssorts();
|
|
|
|
this.createdMergedQuestAssorts = true;
|
|
|
|
}
|
2024-02-05 15:43:46 +01:00
|
|
|
traderClone.assort = this.assortHelper.stripLockedQuestAssort(
|
2023-11-16 02:35:05 +01:00
|
|
|
pmcProfile,
|
|
|
|
traderId,
|
2024-02-05 15:43:46 +01:00
|
|
|
traderClone.assort,
|
2023-11-16 02:35:05 +01:00
|
|
|
this.mergedQuestAssorts,
|
|
|
|
flea,
|
|
|
|
);
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
// Multiply price if multiplier is other than 1
|
|
|
|
if (this.traderConfig.traderPriceMultipler !== 1)
|
|
|
|
{
|
2024-02-05 15:43:46 +01:00
|
|
|
this.multiplyItemPricesByConfigMultiplier(traderClone.assort);
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
2024-02-05 15:43:46 +01:00
|
|
|
return traderClone.assort;
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
2024-01-20 01:19:13 +01:00
|
|
|
/**
|
|
|
|
* Reset every traders root item `BuyRestrictionCurrent` property to 0
|
|
|
|
* @param assortItems Items to adjust
|
|
|
|
*/
|
|
|
|
protected resetBuyRestrictionCurrentValue(assortItems: Item[]): void
|
|
|
|
{
|
|
|
|
// iterate over root items
|
2024-02-02 19:54:07 +01:00
|
|
|
for (const assort of assortItems.filter((item) => item.slotId === "hideout"))
|
2024-01-20 01:19:13 +01:00
|
|
|
{
|
|
|
|
// no value to adjust
|
|
|
|
if (!assort.upd.BuyRestrictionCurrent)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
assort.upd.BuyRestrictionCurrent = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-03 16:23:46 +01:00
|
|
|
/**
|
|
|
|
* Create a dict of all assort id = quest id mappings used to work out what items should be shown to player based on the quests they've started/completed/failed
|
|
|
|
*/
|
|
|
|
protected hydrateMergedQuestAssorts(): void
|
|
|
|
{
|
|
|
|
const traders = this.databaseServer.getTables().traders;
|
|
|
|
|
|
|
|
// Loop every trader
|
|
|
|
for (const traderId in traders)
|
|
|
|
{
|
|
|
|
// Trader has quest assort data
|
|
|
|
const trader = traders[traderId];
|
|
|
|
if (trader.questassort)
|
|
|
|
{
|
|
|
|
// Started/Success/fail
|
|
|
|
for (const questStatus in trader.questassort)
|
|
|
|
{
|
|
|
|
// Each assort to quest id record
|
|
|
|
for (const assortId in trader.questassort[questStatus])
|
|
|
|
{
|
|
|
|
// Null guard
|
|
|
|
if (!this.mergedQuestAssorts[questStatus])
|
|
|
|
{
|
|
|
|
this.mergedQuestAssorts[questStatus] = {};
|
|
|
|
}
|
|
|
|
|
|
|
|
this.mergedQuestAssorts[questStatus][assortId] = trader.questassort[questStatus][assortId];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Reset a traders assorts and move nextResupply value to future
|
|
|
|
* Flag trader as needing a flea offer reset to be picked up by flea update() function
|
|
|
|
* @param trader trader details to alter
|
|
|
|
*/
|
|
|
|
public resetExpiredTrader(trader: ITrader): void
|
|
|
|
{
|
|
|
|
trader.assort.items = this.getPristineTraderAssorts(trader.base._id);
|
|
|
|
|
|
|
|
// Update resupply value to next timestamp
|
|
|
|
trader.base.nextResupply = this.traderHelper.getNextUpdateTimestamp(trader.base._id);
|
|
|
|
|
|
|
|
// Flag a refresh is needed so ragfair update() will pick it up
|
|
|
|
trader.base.refreshTraderRagfairOffers = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Does the supplied trader need its assorts refreshed
|
|
|
|
* @param traderID Trader to check
|
|
|
|
* @returns true they need refreshing
|
|
|
|
*/
|
|
|
|
public traderAssortsHaveExpired(traderID: string): boolean
|
|
|
|
{
|
|
|
|
const time = this.timeUtil.getTimestamp();
|
|
|
|
const trader = this.databaseServer.getTables().traders[traderID];
|
|
|
|
|
|
|
|
return trader.base.nextResupply <= time;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Iterate over all assorts barter_scheme values, find barters selling for money and multiply by multipler in config
|
|
|
|
* @param traderAssort Assorts to multiple price of
|
|
|
|
*/
|
|
|
|
protected multiplyItemPricesByConfigMultiplier(traderAssort: ITraderAssort): void
|
|
|
|
{
|
|
|
|
if (!this.traderConfig.traderPriceMultipler || this.traderConfig.traderPriceMultipler <= 0)
|
|
|
|
{
|
|
|
|
this.traderConfig.traderPriceMultipler = 0.01;
|
|
|
|
this.logger.warning(this.localisationService.getText("trader-price_multipler_is_zero_use_default"));
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const assortId in traderAssort.barter_scheme)
|
|
|
|
{
|
|
|
|
const schemeDetails = traderAssort.barter_scheme[assortId][0];
|
|
|
|
if (schemeDetails.length === 1 && this.paymentHelper.isMoneyTpl(schemeDetails[0]._tpl))
|
|
|
|
{
|
|
|
|
schemeDetails[0].count = Math.ceil(schemeDetails[0].count * this.traderConfig.traderPriceMultipler);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get an array of pristine trader items prior to any alteration by player (as they were on server start)
|
|
|
|
* @param traderId trader id
|
|
|
|
* @returns array of Items
|
|
|
|
*/
|
|
|
|
protected getPristineTraderAssorts(traderId: string): Item[]
|
|
|
|
{
|
|
|
|
return this.jsonUtil.clone(this.traderAssortService.getPristineTraderAssort(traderId).items);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns generated ragfair offers in a trader assort format
|
|
|
|
* @returns Trader assort object
|
|
|
|
*/
|
|
|
|
protected getRagfairDataAsTraderAssort(): ITraderAssort
|
|
|
|
{
|
|
|
|
return {
|
2024-01-23 15:17:01 +01:00
|
|
|
items: this.ragfairAssortGenerator.getAssortItems().flat(),
|
2023-03-03 16:23:46 +01:00
|
|
|
barter_scheme: {},
|
|
|
|
loyal_level_items: {},
|
2023-11-16 02:35:05 +01:00
|
|
|
nextResupply: null,
|
2023-03-03 16:23:46 +01:00
|
|
|
};
|
|
|
|
}
|
2023-11-16 02:35:05 +01:00
|
|
|
}
|