2023-03-03 15:23:46 +00:00
|
|
|
import { inject, injectable } from "tsyringe";
|
2024-05-21 17:59:04 +00:00
|
|
|
import { ProfileHelper } from "@spt/helpers/ProfileHelper";
|
|
|
|
import { IPmcData } from "@spt/models/eft/common/IPmcData";
|
|
|
|
import { IHideoutImprovement, Productive, TraderInfo } from "@spt/models/eft/common/tables/IBotBase";
|
|
|
|
import { ProfileChange, TraderData } from "@spt/models/eft/itemEvent/IItemEventRouterBase";
|
|
|
|
import { IItemEventRouterResponse } from "@spt/models/eft/itemEvent/IItemEventRouterResponse";
|
|
|
|
import { ICloner } from "@spt/utils/cloners/ICloner";
|
|
|
|
import { TimeUtil } from "@spt/utils/TimeUtil";
|
2023-03-03 15:23:46 +00:00
|
|
|
|
|
|
|
@injectable()
|
|
|
|
export class EventOutputHolder
|
|
|
|
{
|
2024-05-31 23:10:18 +01:00
|
|
|
/**
|
|
|
|
* What has client been informed of this game session
|
|
|
|
* Key = sessionId, then second key is prod id
|
|
|
|
*/
|
|
|
|
protected clientActiveSessionStorage: Record<string, Record<string, { clientInformed: boolean }>> = {};
|
2023-03-03 15:23:46 +00:00
|
|
|
|
|
|
|
constructor(
|
|
|
|
@inject("ProfileHelper") protected profileHelper: ProfileHelper,
|
2023-11-16 21:42:06 +00:00
|
|
|
@inject("TimeUtil") protected timeUtil: TimeUtil,
|
2024-05-28 14:04:20 +00:00
|
|
|
@inject("PrimaryCloner") protected cloner: ICloner,
|
2023-03-03 15:23:46 +00:00
|
|
|
)
|
|
|
|
{}
|
|
|
|
|
|
|
|
// TODO REMEMBER TO CHANGE OUTPUT
|
2023-11-16 21:42:06 +00:00
|
|
|
protected output: IItemEventRouterResponse = { warnings: [], profileChanges: {} };
|
2023-03-03 15:23:46 +00:00
|
|
|
|
|
|
|
public getOutput(sessionID: string): IItemEventRouterResponse
|
|
|
|
{
|
|
|
|
if (!this.output.profileChanges[sessionID])
|
|
|
|
{
|
|
|
|
this.resetOutput(sessionID);
|
|
|
|
}
|
|
|
|
|
|
|
|
return this.output;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Reset the response object to a default state
|
|
|
|
* Occurs prior to event being handled by server
|
|
|
|
* @param sessionID Players id
|
|
|
|
*/
|
|
|
|
public resetOutput(sessionID: string): void
|
|
|
|
{
|
|
|
|
const pmcData: IPmcData = this.profileHelper.getPmcProfile(sessionID);
|
|
|
|
|
|
|
|
this.output.warnings = [];
|
|
|
|
this.output.profileChanges[sessionID] = {
|
|
|
|
_id: sessionID,
|
|
|
|
experience: pmcData.Info.Experience,
|
|
|
|
quests: [],
|
|
|
|
ragFairOffers: [],
|
2023-10-10 11:03:20 +00:00
|
|
|
weaponBuilds: [],
|
|
|
|
equipmentBuilds: [],
|
2023-11-16 21:42:06 +00:00
|
|
|
items: { new: [], change: [], del: [] },
|
2023-03-03 15:23:46 +00:00
|
|
|
production: {},
|
|
|
|
improvements: {},
|
2023-11-16 21:42:06 +00:00
|
|
|
skills: { Common: [], Mastering: [], Points: 0 },
|
2024-05-13 17:58:17 +00:00
|
|
|
health: this.cloner.clone(pmcData.Health),
|
2023-03-03 15:23:46 +00:00
|
|
|
traderRelations: {},
|
2023-11-16 21:42:06 +00:00
|
|
|
// changedHideoutStashes: {},
|
2023-03-03 15:23:46 +00:00
|
|
|
recipeUnlocked: {},
|
2023-11-16 21:42:06 +00:00
|
|
|
questsStatus: [],
|
2023-03-03 15:23:46 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Update output object with most recent values from player profile
|
|
|
|
* @param sessionId Session id
|
|
|
|
*/
|
|
|
|
public updateOutputProperties(sessionId: string): void
|
|
|
|
{
|
|
|
|
const pmcData: IPmcData = this.profileHelper.getPmcProfile(sessionId);
|
|
|
|
const profileChanges: ProfileChange = this.output.profileChanges[sessionId];
|
|
|
|
|
|
|
|
profileChanges.experience = pmcData.Info.Experience;
|
2024-05-13 17:58:17 +00:00
|
|
|
profileChanges.health = this.cloner.clone(pmcData.Health);
|
|
|
|
profileChanges.skills.Common = this.cloner.clone(pmcData.Skills.Common); // Always send skills for Item event route response
|
|
|
|
profileChanges.skills.Mastering = this.cloner.clone(pmcData.Skills.Mastering);
|
2023-10-10 11:03:20 +00:00
|
|
|
|
|
|
|
// Clone productions to ensure we preseve the profile jsons data
|
2023-11-16 21:42:06 +00:00
|
|
|
profileChanges.production = this.getProductionsFromProfileAndFlagComplete(
|
2024-05-13 17:58:17 +00:00
|
|
|
this.cloner.clone(pmcData.Hideout.Production),
|
2024-05-31 23:10:18 +01:00
|
|
|
sessionId,
|
2023-11-16 21:42:06 +00:00
|
|
|
);
|
2024-05-13 17:58:17 +00:00
|
|
|
profileChanges.improvements = this.cloner.clone(this.getImprovementsFromProfileAndFlagComplete(pmcData));
|
2023-10-10 11:03:20 +00:00
|
|
|
profileChanges.traderRelations = this.constructTraderRelations(pmcData.TradersInfo);
|
2023-11-16 12:55:57 +00:00
|
|
|
|
2023-11-20 10:14:21 +00:00
|
|
|
// Fixes container craft from water collector not resetting after collection + removed completed normal crafts
|
|
|
|
this.cleanUpCompleteCraftsInProfile(pmcData.Hideout.Production);
|
2023-10-10 11:03:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Convert the internal trader data object into an object we can send to the client
|
|
|
|
* @param traderData server data for traders
|
|
|
|
* @returns dict of trader id + TraderData
|
|
|
|
*/
|
|
|
|
protected constructTraderRelations(traderData: Record<string, TraderInfo>): Record<string, TraderData>
|
|
|
|
{
|
|
|
|
const result: Record<string, TraderData> = {};
|
|
|
|
|
|
|
|
for (const traderId in traderData)
|
|
|
|
{
|
|
|
|
const baseData = traderData[traderId];
|
|
|
|
result[traderId] = {
|
|
|
|
salesSum: baseData.salesSum,
|
|
|
|
disabled: baseData.disabled,
|
|
|
|
loyalty: baseData.loyaltyLevel,
|
|
|
|
standing: baseData.standing,
|
2023-11-16 21:42:06 +00:00
|
|
|
unlocked: baseData.unlocked,
|
2023-10-10 11:03:20 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|
2023-11-16 21:42:06 +00:00
|
|
|
|
2023-03-03 15:23:46 +00:00
|
|
|
/**
|
|
|
|
* Return all hideout Improvements from player profile, adjust completed Improvements' completed property to be true
|
|
|
|
* @param pmcData Player profile
|
|
|
|
* @returns dictionary of hideout improvements
|
|
|
|
*/
|
|
|
|
protected getImprovementsFromProfileAndFlagComplete(pmcData: IPmcData): Record<string, IHideoutImprovement>
|
|
|
|
{
|
2023-10-10 11:03:20 +00:00
|
|
|
for (const improvementKey in pmcData.Hideout.Improvement)
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
2023-10-10 11:03:20 +00:00
|
|
|
const improvement = pmcData.Hideout.Improvement[improvementKey];
|
2023-03-03 15:23:46 +00:00
|
|
|
|
|
|
|
// Skip completed
|
|
|
|
if (improvement.completed)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (improvement.improveCompleteTimestamp < this.timeUtil.getTimestamp())
|
|
|
|
{
|
|
|
|
improvement.completed = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-10-10 11:03:20 +00:00
|
|
|
return pmcData.Hideout.Improvement;
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2023-10-10 11:03:20 +00:00
|
|
|
* Return productions from player profile except those completed crafts the client has already seen
|
2023-03-03 15:23:46 +00:00
|
|
|
* @param pmcData Player profile
|
|
|
|
* @returns dictionary of hideout productions
|
|
|
|
*/
|
2023-11-16 21:42:06 +00:00
|
|
|
protected getProductionsFromProfileAndFlagComplete(
|
|
|
|
productions: Record<string, Productive>,
|
2024-05-31 23:10:18 +01:00
|
|
|
sessionId: string,
|
2024-05-27 20:06:07 +00:00
|
|
|
): Record<string, Productive> | undefined
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
|
|
|
for (const productionKey in productions)
|
|
|
|
{
|
|
|
|
const production = productions[productionKey];
|
2023-10-28 17:48:37 +01:00
|
|
|
if (!production)
|
|
|
|
{
|
|
|
|
// Could be cancelled production, skip item to save processing
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2023-11-20 10:14:21 +00:00
|
|
|
// Complete and is Continuous e.g. water collector
|
|
|
|
if (production.sptIsComplete && production.sptIsContinuous)
|
2023-11-16 12:55:57 +00:00
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2023-10-28 17:48:37 +01:00
|
|
|
// Skip completed
|
2023-03-03 15:23:46 +00:00
|
|
|
if (!production.inProgress)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2023-10-10 11:03:20 +00:00
|
|
|
// Client informed of craft, remove from data returned
|
2024-05-31 23:10:18 +01:00
|
|
|
const storageForSessionId = this.clientActiveSessionStorage[sessionId];
|
|
|
|
if (storageForSessionId[productionKey]?.clientInformed)
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
|
|
|
delete productions[productionKey];
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2023-10-10 11:03:20 +00:00
|
|
|
// Flag started craft as having been seen by client
|
2024-05-31 23:10:18 +01:00
|
|
|
if (production.Progress > 0 && !storageForSessionId[productionKey]?.clientInformed)
|
2023-04-23 19:24:00 +01:00
|
|
|
{
|
2024-05-31 23:10:18 +01:00
|
|
|
storageForSessionId[productionKey] = { clientInformed: true };
|
2023-04-23 19:24:00 +01:00
|
|
|
}
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|
|
|
|
|
2024-05-27 20:06:07 +00:00
|
|
|
// Return undefined if there's no crafts to send to client to match live behaviour
|
|
|
|
return Object.keys(productions).length > 0 ? productions : undefined;
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|
2023-11-16 12:55:57 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Required as continuous productions don't reset and stay at 100% completion but client thinks it hasn't started
|
|
|
|
* @param productions Productions in a profile
|
|
|
|
*/
|
2023-11-20 10:14:21 +00:00
|
|
|
protected cleanUpCompleteCraftsInProfile(productions: Record<string, Productive>): void
|
2023-11-16 12:55:57 +00:00
|
|
|
{
|
|
|
|
for (const productionKey in productions)
|
|
|
|
{
|
|
|
|
const production = productions[productionKey];
|
2023-12-10 22:31:55 +00:00
|
|
|
if (production?.sptIsComplete && production?.sptIsContinuous)
|
2023-11-16 12:55:57 +00:00
|
|
|
{
|
2023-11-20 10:14:21 +00:00
|
|
|
// Water collector / Bitcoin etc
|
2023-11-16 12:55:57 +00:00
|
|
|
production.sptIsComplete = false;
|
|
|
|
production.Progress = 0;
|
|
|
|
production.StartTimestamp = this.timeUtil.getTimestamp();
|
|
|
|
}
|
2023-12-10 22:31:55 +00:00
|
|
|
else if (!production?.inProgress)
|
2023-11-20 10:14:21 +00:00
|
|
|
{
|
|
|
|
// Normal completed craft, delete
|
|
|
|
delete productions[productionKey];
|
|
|
|
}
|
2023-11-16 12:55:57 +00:00
|
|
|
}
|
|
|
|
}
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|