2023-03-03 15:23:46 +00:00
|
|
|
import { inject, injectable } from "tsyringe";
|
2024-05-21 17:59:04 +00:00
|
|
|
import { Item } from "@spt/models/eft/common/tables/IItem";
|
|
|
|
import { ITemplateItem, Props } from "@spt/models/eft/common/tables/ITemplateItem";
|
|
|
|
import { ConfigTypes } from "@spt/models/enums/ConfigTypes";
|
|
|
|
import { IRepairConfig } from "@spt/models/spt/config/IRepairConfig";
|
|
|
|
import { ILogger } from "@spt/models/spt/utils/ILogger";
|
|
|
|
import { ConfigServer } from "@spt/servers/ConfigServer";
|
|
|
|
import { DatabaseServer } from "@spt/servers/DatabaseServer";
|
|
|
|
import { ICloner } from "@spt/utils/cloners/ICloner";
|
|
|
|
import { RandomUtil } from "@spt/utils/RandomUtil";
|
2023-03-03 15:23:46 +00:00
|
|
|
|
|
|
|
@injectable()
|
|
|
|
export class RepairHelper
|
|
|
|
{
|
|
|
|
protected repairConfig: IRepairConfig;
|
|
|
|
|
|
|
|
constructor(
|
|
|
|
@inject("WinstonLogger") protected logger: ILogger,
|
|
|
|
@inject("RandomUtil") protected randomUtil: RandomUtil,
|
|
|
|
@inject("DatabaseServer") protected databaseServer: DatabaseServer,
|
2023-11-16 21:42:06 +00:00
|
|
|
@inject("ConfigServer") protected configServer: ConfigServer,
|
2024-05-13 17:58:17 +00:00
|
|
|
@inject("RecursiveCloner") protected cloner: ICloner,
|
2023-03-03 15:23:46 +00:00
|
|
|
)
|
|
|
|
{
|
|
|
|
this.repairConfig = this.configServer.getConfig(ConfigTypes.REPAIR);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Alter an items durability after a repair by trader/repair kit
|
|
|
|
* @param itemToRepair item to update durability details
|
|
|
|
* @param itemToRepairDetails db details of item to repair
|
|
|
|
* @param isArmor Is item being repaired a piece of armor
|
|
|
|
* @param amountToRepair how many unit of durability to repair
|
|
|
|
* @param useRepairKit Is item being repaired with a repair kit
|
2023-10-10 11:03:20 +00:00
|
|
|
* @param traderQualityMultipler Trader quality value from traders base json
|
2023-03-03 15:23:46 +00:00
|
|
|
* @param applyMaxDurabilityDegradation should item have max durability reduced
|
|
|
|
*/
|
|
|
|
public updateItemDurability(
|
|
|
|
itemToRepair: Item,
|
|
|
|
itemToRepairDetails: ITemplateItem,
|
|
|
|
isArmor: boolean,
|
|
|
|
amountToRepair: number,
|
|
|
|
useRepairKit: boolean,
|
|
|
|
traderQualityMultipler: number,
|
2023-11-16 21:42:06 +00:00
|
|
|
applyMaxDurabilityDegradation = true,
|
2023-03-03 15:23:46 +00:00
|
|
|
): void
|
|
|
|
{
|
|
|
|
this.logger.debug(`Adding ${amountToRepair} to ${itemToRepairDetails._name} using kit: ${useRepairKit}`);
|
2023-12-14 15:47:01 +00:00
|
|
|
|
2024-05-13 17:58:17 +00:00
|
|
|
const itemMaxDurability = this.cloner.clone(itemToRepair.upd.Repairable.MaxDurability);
|
|
|
|
const itemCurrentDurability = this.cloner.clone(itemToRepair.upd.Repairable.Durability);
|
|
|
|
const itemCurrentMaxDurability = this.cloner.clone(itemToRepair.upd.Repairable.MaxDurability);
|
2023-03-03 15:23:46 +00:00
|
|
|
|
|
|
|
let newCurrentDurability = itemCurrentDurability + amountToRepair;
|
|
|
|
let newCurrentMaxDurability = itemCurrentMaxDurability + amountToRepair;
|
|
|
|
|
|
|
|
// Ensure new max isnt above items max
|
|
|
|
if (newCurrentMaxDurability > itemMaxDurability)
|
|
|
|
{
|
|
|
|
newCurrentMaxDurability = itemMaxDurability;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure new current isnt above items max
|
|
|
|
if (newCurrentDurability > itemMaxDurability)
|
|
|
|
{
|
|
|
|
newCurrentDurability = itemMaxDurability;
|
|
|
|
}
|
|
|
|
|
2024-01-08 21:17:23 +00:00
|
|
|
// Update Repairable properties with new values after repair
|
2023-11-16 21:42:06 +00:00
|
|
|
itemToRepair.upd.Repairable = { Durability: newCurrentDurability, MaxDurability: newCurrentMaxDurability };
|
2023-03-03 15:23:46 +00:00
|
|
|
|
|
|
|
// when modders set the repair coefficient to 0 it means that they dont want to lose durability on items
|
|
|
|
// the code below generates a random degradation on the weapon durability
|
|
|
|
if (applyMaxDurabilityDegradation)
|
|
|
|
{
|
2023-11-16 21:42:06 +00:00
|
|
|
const randomisedWearAmount = isArmor
|
|
|
|
? this.getRandomisedArmorRepairDegradationValue(
|
|
|
|
itemToRepairDetails._props.ArmorMaterial,
|
|
|
|
useRepairKit,
|
|
|
|
itemCurrentMaxDurability,
|
|
|
|
traderQualityMultipler,
|
|
|
|
)
|
|
|
|
: this.getRandomisedWeaponRepairDegradationValue(
|
|
|
|
itemToRepairDetails._props,
|
|
|
|
useRepairKit,
|
|
|
|
itemCurrentMaxDurability,
|
|
|
|
traderQualityMultipler,
|
|
|
|
);
|
|
|
|
|
2023-03-03 15:23:46 +00:00
|
|
|
// Apply wear to durability
|
|
|
|
itemToRepair.upd.Repairable.MaxDurability -= randomisedWearAmount;
|
2023-11-16 21:42:06 +00:00
|
|
|
|
2023-03-03 15:23:46 +00:00
|
|
|
// After adjusting max durability with degradation, ensure current dura isnt above max
|
|
|
|
if (itemToRepair.upd.Repairable.Durability > itemToRepair.upd.Repairable.MaxDurability)
|
|
|
|
{
|
|
|
|
itemToRepair.upd.Repairable.Durability = itemToRepair.upd.Repairable.MaxDurability;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Repair mask cracks
|
2024-01-08 21:17:23 +00:00
|
|
|
if (itemToRepair.upd.FaceShield && itemToRepair.upd.FaceShield?.Hits > 0)
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
|
|
|
itemToRepair.upd.FaceShield.Hits = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-12-14 15:47:01 +00:00
|
|
|
/**
|
|
|
|
* Repairing armor reduces the total durability value slightly, get a randomised (to 2dp) amount based on armor material
|
|
|
|
* @param armorMaterial What material is the armor being repaired made of
|
|
|
|
* @param isRepairKit Was a repair kit used
|
|
|
|
* @param armorMax Max amount of durability item can have
|
|
|
|
* @param traderQualityMultipler Different traders produce different loss values
|
|
|
|
* @returns Amount to reduce max durability by
|
|
|
|
*/
|
2023-11-16 21:42:06 +00:00
|
|
|
protected getRandomisedArmorRepairDegradationValue(
|
|
|
|
armorMaterial: string,
|
|
|
|
isRepairKit: boolean,
|
|
|
|
armorMax: number,
|
|
|
|
traderQualityMultipler: number,
|
|
|
|
): number
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
2023-12-14 15:47:01 +00:00
|
|
|
// Degradation value is based on the armor material
|
2024-05-28 10:25:23 +01:00
|
|
|
const armorMaterialSettings = this.databaseServer.getTables().globals!.config.ArmorMaterials[armorMaterial];
|
2023-03-03 15:23:46 +00:00
|
|
|
|
|
|
|
const minMultiplier = isRepairKit
|
|
|
|
? armorMaterialSettings.MinRepairKitDegradation
|
|
|
|
: armorMaterialSettings.MinRepairDegradation;
|
|
|
|
|
|
|
|
const maxMultiplier = isRepairKit
|
|
|
|
? armorMaterialSettings.MaxRepairKitDegradation
|
|
|
|
: armorMaterialSettings.MaxRepairDegradation;
|
|
|
|
|
|
|
|
const duraLossPercent = this.randomUtil.getFloat(minMultiplier, maxMultiplier);
|
2024-05-07 23:57:08 -04:00
|
|
|
const duraLossMultipliedByTraderMultiplier = duraLossPercent * armorMax * traderQualityMultipler;
|
2023-03-03 15:23:46 +00:00
|
|
|
|
|
|
|
return Number(duraLossMultipliedByTraderMultiplier.toFixed(2));
|
|
|
|
}
|
|
|
|
|
2023-12-14 15:47:01 +00:00
|
|
|
/**
|
|
|
|
* Repairing weapons reduces the total durability value slightly, get a randomised (to 2dp) amount
|
|
|
|
* @param itemProps Weapon properties
|
|
|
|
* @param isRepairKit Was a repair kit used
|
|
|
|
* @param weaponMax ax amount of durability item can have
|
|
|
|
* @param traderQualityMultipler Different traders produce different loss values
|
|
|
|
* @returns Amount to reduce max durability by
|
|
|
|
*/
|
2023-11-16 21:42:06 +00:00
|
|
|
protected getRandomisedWeaponRepairDegradationValue(
|
|
|
|
itemProps: Props,
|
|
|
|
isRepairKit: boolean,
|
|
|
|
weaponMax: number,
|
|
|
|
traderQualityMultipler: number,
|
|
|
|
): number
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
2023-11-16 21:42:06 +00:00
|
|
|
const minRepairDeg = isRepairKit ? itemProps.MinRepairKitDegradation : itemProps.MinRepairDegradation;
|
|
|
|
let maxRepairDeg = isRepairKit ? itemProps.MaxRepairKitDegradation : itemProps.MaxRepairDegradation;
|
2023-03-03 15:23:46 +00:00
|
|
|
|
|
|
|
// WORKAROUND: Some items are always 0 when repairkit is true
|
|
|
|
if (maxRepairDeg === 0)
|
|
|
|
{
|
|
|
|
maxRepairDeg = itemProps.MaxRepairDegradation;
|
|
|
|
}
|
|
|
|
|
|
|
|
const duraLossPercent = this.randomUtil.getFloat(minRepairDeg, maxRepairDeg);
|
2024-05-07 23:57:08 -04:00
|
|
|
const duraLossMultipliedByTraderMultiplier = duraLossPercent * weaponMax * traderQualityMultipler;
|
2023-03-03 15:23:46 +00:00
|
|
|
|
|
|
|
return Number(duraLossMultipliedByTraderMultiplier.toFixed(2));
|
|
|
|
}
|
2023-11-16 21:42:06 +00:00
|
|
|
}
|