2023-03-03 15:23:46 +00:00
|
|
|
import { inject, injectable } from "tsyringe";
|
|
|
|
|
2023-10-19 17:21:17 +00:00
|
|
|
import { IPmcData } from "@spt-aki/models/eft/common/IPmcData";
|
|
|
|
import { ISyncHealthRequestData } from "@spt-aki/models/eft/health/ISyncHealthRequestData";
|
|
|
|
import { Effects, IAkiProfile } from "@spt-aki/models/eft/profile/IAkiProfile";
|
|
|
|
import { ConfigTypes } from "@spt-aki/models/enums/ConfigTypes";
|
|
|
|
import { IHealthConfig } from "@spt-aki/models/spt/config/IHealthConfig";
|
|
|
|
import { ILogger } from "@spt-aki/models/spt/utils/ILogger";
|
|
|
|
import { ConfigServer } from "@spt-aki/servers/ConfigServer";
|
|
|
|
import { SaveServer } from "@spt-aki/servers/SaveServer";
|
|
|
|
import { JsonUtil } from "@spt-aki/utils/JsonUtil";
|
|
|
|
import { TimeUtil } from "@spt-aki/utils/TimeUtil";
|
2023-03-03 15:23:46 +00:00
|
|
|
|
|
|
|
@injectable()
|
|
|
|
export class HealthHelper
|
|
|
|
{
|
|
|
|
protected healthConfig: IHealthConfig;
|
|
|
|
|
|
|
|
constructor(
|
|
|
|
@inject("JsonUtil") protected jsonUtil: JsonUtil,
|
|
|
|
@inject("WinstonLogger") protected logger: ILogger,
|
|
|
|
@inject("TimeUtil") protected timeUtil: TimeUtil,
|
|
|
|
@inject("SaveServer") protected saveServer: SaveServer,
|
2023-11-15 20:35:05 -05:00
|
|
|
@inject("ConfigServer") protected configServer: ConfigServer,
|
2023-03-03 15:23:46 +00:00
|
|
|
)
|
|
|
|
{
|
|
|
|
this.healthConfig = this.configServer.getConfig(ConfigTypes.HEALTH);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Resets the profiles vitality/health and vitality/effects properties to their defaults
|
|
|
|
* @param sessionID Session Id
|
|
|
|
* @returns updated profile
|
|
|
|
*/
|
|
|
|
public resetVitality(sessionID: string): IAkiProfile
|
|
|
|
{
|
|
|
|
const profile = this.saveServer.getProfile(sessionID);
|
|
|
|
|
2023-11-15 20:35:05 -05:00
|
|
|
if (!profile.vitality)
|
|
|
|
{ // Occurs on newly created profiles
|
|
|
|
profile.vitality = { health: null, effects: null };
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|
|
|
|
profile.vitality.health = {
|
|
|
|
Hydration: 0,
|
|
|
|
Energy: 0,
|
|
|
|
Temperature: 0,
|
|
|
|
Head: 0,
|
|
|
|
Chest: 0,
|
|
|
|
Stomach: 0,
|
|
|
|
LeftArm: 0,
|
|
|
|
RightArm: 0,
|
|
|
|
LeftLeg: 0,
|
2023-11-15 20:35:05 -05:00
|
|
|
RightLeg: 0,
|
2023-03-03 15:23:46 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
profile.vitality.effects = {
|
|
|
|
Head: {},
|
|
|
|
Chest: {},
|
|
|
|
Stomach: {},
|
|
|
|
LeftArm: {},
|
|
|
|
RightArm: {},
|
|
|
|
LeftLeg: {},
|
2023-11-15 20:35:05 -05:00
|
|
|
RightLeg: {},
|
2023-03-03 15:23:46 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
return profile;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Update player profile with changes from request object
|
|
|
|
* @param pmcData Player profile
|
|
|
|
* @param request Heal request
|
|
|
|
* @param sessionID Session id
|
|
|
|
* @param addEffects Should effects be added or removed (default - add)
|
2023-10-10 11:03:20 +00:00
|
|
|
* @param deleteExistingEffects Should all prior effects be removed before apply new ones
|
2023-03-03 15:23:46 +00:00
|
|
|
*/
|
2023-11-15 20:35:05 -05:00
|
|
|
public saveVitality(
|
|
|
|
pmcData: IPmcData,
|
|
|
|
request: ISyncHealthRequestData,
|
|
|
|
sessionID: string,
|
|
|
|
addEffects = true,
|
|
|
|
deleteExistingEffects = true,
|
|
|
|
): void
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
|
|
|
const postRaidBodyParts = request.Health; // post raid health settings
|
|
|
|
const profile = this.saveServer.getProfile(sessionID);
|
|
|
|
const profileHealth = profile.vitality.health;
|
|
|
|
const profileEffects = profile.vitality.effects;
|
|
|
|
|
|
|
|
profileHealth.Hydration = request.Hydration;
|
|
|
|
profileHealth.Energy = request.Energy;
|
|
|
|
profileHealth.Temperature = request.Temperature;
|
|
|
|
|
|
|
|
// Transfer properties from request to profile
|
|
|
|
for (const bodyPart in postRaidBodyParts)
|
|
|
|
{
|
|
|
|
// Transfer effects from request to profile
|
|
|
|
if (postRaidBodyParts[bodyPart].Effects)
|
|
|
|
{
|
|
|
|
profileEffects[bodyPart] = postRaidBodyParts[bodyPart].Effects;
|
|
|
|
}
|
2023-11-15 20:35:05 -05:00
|
|
|
if (request.IsAlive === true)
|
|
|
|
{ // is player alive, not is limb alive
|
2023-03-03 15:23:46 +00:00
|
|
|
profileHealth[bodyPart] = postRaidBodyParts[bodyPart].Current;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-11-15 20:35:05 -05:00
|
|
|
profileHealth[bodyPart] = pmcData.Health.BodyParts[bodyPart].Health.Maximum
|
|
|
|
* this.healthConfig.healthMultipliers.death;
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add effects to body parts
|
|
|
|
if (addEffects)
|
|
|
|
{
|
2023-11-15 20:35:05 -05:00
|
|
|
this.saveEffects(
|
|
|
|
pmcData,
|
|
|
|
sessionID,
|
|
|
|
this.jsonUtil.clone(this.saveServer.getProfile(sessionID).vitality.effects),
|
|
|
|
deleteExistingEffects,
|
|
|
|
);
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Adjust hydration/energy/temp and limb hp
|
|
|
|
this.saveHealth(pmcData, sessionID);
|
|
|
|
|
|
|
|
this.resetVitality(sessionID);
|
|
|
|
|
|
|
|
pmcData.Health.UpdateTime = this.timeUtil.getTimestamp();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Adjust hydration/energy/temperate and body part hp values in player profile to values in profile.vitality
|
|
|
|
* @param pmcData Profile to update
|
|
|
|
* @param sessionId Session id
|
|
|
|
*/
|
|
|
|
protected saveHealth(pmcData: IPmcData, sessionID: string): void
|
|
|
|
{
|
|
|
|
if (!this.healthConfig.save.health)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const profileHealth = this.saveServer.getProfile(sessionID).vitality.health;
|
|
|
|
for (const healthModifier in profileHealth)
|
|
|
|
{
|
|
|
|
let target = profileHealth[healthModifier];
|
|
|
|
|
|
|
|
if (["Hydration", "Energy", "Temperature"].includes(healthModifier))
|
|
|
|
{
|
|
|
|
// Set resources
|
|
|
|
if (target > pmcData.Health[healthModifier].Maximum)
|
|
|
|
{
|
|
|
|
target = pmcData.Health[healthModifier].Maximum;
|
|
|
|
}
|
|
|
|
|
|
|
|
pmcData.Health[healthModifier].Current = Math.round(target);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Over max, limit
|
|
|
|
if (target > pmcData.Health.BodyParts[healthModifier].Health.Maximum)
|
|
|
|
{
|
|
|
|
target = pmcData.Health.BodyParts[healthModifier].Health.Maximum;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Part was zeroed out in raid
|
|
|
|
if (target === 0)
|
|
|
|
{
|
|
|
|
// Blacked body part
|
2023-11-15 20:35:05 -05:00
|
|
|
target = Math.round(
|
|
|
|
pmcData.Health.BodyParts[healthModifier].Health.Maximum
|
|
|
|
* this.healthConfig.healthMultipliers.blacked,
|
|
|
|
);
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pmcData.Health.BodyParts[healthModifier].Health.Current = Math.round(target);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Save effects to profile
|
|
|
|
* Works by removing all effects and adding them back from profile
|
|
|
|
* Removes empty 'Effects' objects if found
|
|
|
|
* @param pmcData Player profile
|
|
|
|
* @param sessionId Session id
|
|
|
|
* @param bodyPartsWithEffects dict of body parts with effects that should be added to profile
|
|
|
|
* @param addEffects Should effects be added back to profile
|
|
|
|
*/
|
2023-11-15 20:35:05 -05:00
|
|
|
protected saveEffects(
|
|
|
|
pmcData: IPmcData,
|
|
|
|
sessionId: string,
|
|
|
|
bodyPartsWithEffects: Effects,
|
|
|
|
deleteExistingEffects = true,
|
|
|
|
): void
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
|
|
|
if (!this.healthConfig.save.effects)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const bodyPart in bodyPartsWithEffects)
|
|
|
|
{
|
|
|
|
// clear effects from profile bodyPart
|
|
|
|
if (deleteExistingEffects)
|
|
|
|
{
|
|
|
|
delete pmcData.Health.BodyParts[bodyPart].Effects;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const effectType in bodyPartsWithEffects[bodyPart])
|
|
|
|
{
|
|
|
|
if (typeof effectType !== "string")
|
|
|
|
{
|
|
|
|
this.logger.warning(`Effect ${effectType} on body part ${bodyPart} not a string, report this`);
|
|
|
|
}
|
|
|
|
|
|
|
|
// // data can be index or the effect string (e.g. "Fracture") itself
|
|
|
|
// const effect = /^-?\d+$/.test(effectValue) // is an int
|
|
|
|
// ? nodeEffects[bodyPart][effectValue]
|
|
|
|
// : effectValue;
|
|
|
|
let time = bodyPartsWithEffects[bodyPart][effectType];
|
|
|
|
if (time)
|
|
|
|
{
|
|
|
|
// Sometimes the value can be Infinity instead of -1, blame HealthListener.cs in modules
|
|
|
|
if (time === "Infinity")
|
|
|
|
{
|
2023-11-15 20:35:05 -05:00
|
|
|
this.logger.warning(
|
|
|
|
`Effect ${effectType} found with value of Infinity, changed to -1, this is an issue with HealthListener.cs`,
|
|
|
|
);
|
2023-03-03 15:23:46 +00:00
|
|
|
time = -1;
|
|
|
|
}
|
|
|
|
this.addEffect(pmcData, bodyPart, effectType, time);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
this.addEffect(pmcData, bodyPart, effectType);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add effect to body part in profile
|
|
|
|
* @param pmcData Player profile
|
|
|
|
* @param effectBodyPart body part to edit
|
|
|
|
* @param effectType Effect to add to body part
|
|
|
|
* @param duration How long the effect has left in seconds (-1 by default, no duration).
|
|
|
|
*/
|
|
|
|
protected addEffect(pmcData: IPmcData, effectBodyPart: string, effectType: string, duration = -1): void
|
|
|
|
{
|
|
|
|
const profileBodyPart = pmcData.Health.BodyParts[effectBodyPart];
|
|
|
|
if (!profileBodyPart.Effects)
|
|
|
|
{
|
|
|
|
profileBodyPart.Effects = {};
|
|
|
|
}
|
|
|
|
|
|
|
|
profileBodyPart.Effects[effectType] = { Time: duration };
|
|
|
|
|
|
|
|
// Delete empty property to prevent client bugs
|
|
|
|
if (this.isEmpty(profileBodyPart.Effects))
|
|
|
|
{
|
|
|
|
delete profileBodyPart.Effects;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-11-15 20:35:05 -05:00
|
|
|
protected isEmpty(map: Record<string, { Time: number; }>): boolean
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
|
|
|
for (const key in map)
|
|
|
|
{
|
|
|
|
if (key in map)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2023-11-15 20:35:05 -05:00
|
|
|
}
|