Server/project/src/helpers/ItemHelper.ts

1180 lines
41 KiB
TypeScript
Raw Normal View History

2023-03-03 16:23:46 +01:00
import { inject, injectable } from "tsyringe";
import { HandbookHelper } from "@spt-aki/helpers/HandbookHelper";
import { IPmcData } from "@spt-aki/models/eft/common/IPmcData";
import { InsuredItem } from "@spt-aki/models/eft/common/tables/IBotBase";
import { Item, Location, Repairable } from "@spt-aki/models/eft/common/tables/IItem";
import { IStaticAmmoDetails } from "@spt-aki/models/eft/common/tables/ILootBase";
import { ITemplateItem } from "@spt-aki/models/eft/common/tables/ITemplateItem";
import { BaseClasses } from "@spt-aki/models/enums/BaseClasses";
import { ILogger } from "@spt-aki/models/spt/utils/ILogger";
import { DatabaseServer } from "@spt-aki/servers/DatabaseServer";
import { ItemBaseClassService } from "@spt-aki/services/ItemBaseClassService";
import { ItemFilterService } from "@spt-aki/services/ItemFilterService";
import { LocaleService } from "@spt-aki/services/LocaleService";
import { LocalisationService } from "@spt-aki/services/LocalisationService";
import { HashUtil } from "@spt-aki/utils/HashUtil";
import { JsonUtil } from "@spt-aki/utils/JsonUtil";
import { MathUtil } from "@spt-aki/utils/MathUtil";
import { ObjectId } from "@spt-aki/utils/ObjectId";
import { ProbabilityObject, ProbabilityObjectArray, RandomUtil } from "@spt-aki/utils/RandomUtil";
2023-03-03 16:23:46 +01:00
@injectable()
class ItemHelper
{
protected readonly defaultInvalidBaseTypes: string[] = [
BaseClasses.LOOT_CONTAINER,
BaseClasses.MOB_CONTAINER,
BaseClasses.STASH,
BaseClasses.SORTING_TABLE,
BaseClasses.INVENTORY,
BaseClasses.STATIONARY_CONTAINER,
2023-11-13 17:07:59 +01:00
BaseClasses.POCKETS,
];
2023-03-03 16:23:46 +01:00
constructor(
@inject("WinstonLogger") protected logger: ILogger,
@inject("HashUtil") protected hashUtil: HashUtil,
@inject("JsonUtil") protected jsonUtil: JsonUtil,
@inject("RandomUtil") protected randomUtil: RandomUtil,
@inject("ObjectId") protected objectId: ObjectId,
@inject("MathUtil") protected mathUtil: MathUtil,
@inject("DatabaseServer") protected databaseServer: DatabaseServer,
@inject("HandbookHelper") protected handbookHelper: HandbookHelper,
@inject("ItemBaseClassService") protected itemBaseClassService: ItemBaseClassService,
@inject("ItemFilterService") protected itemFilterService: ItemFilterService,
2023-03-03 16:23:46 +01:00
@inject("LocalisationService") protected localisationService: LocalisationService,
2023-11-13 17:07:59 +01:00
@inject("LocaleService") protected localeService: LocaleService,
2023-03-03 16:23:46 +01:00
)
{}
/**
* Checks if an id is a valid item. Valid meaning that it's an item that be stored in stash
* @param {string} tpl the template id / tpl
* @returns boolean; true for items that may be in player possession and not quest items
*/
public isValidItem(tpl: string, invalidBaseTypes: string[] = null): boolean
{
if (invalidBaseTypes === null)
{
invalidBaseTypes = this.defaultInvalidBaseTypes;
2023-03-03 16:23:46 +01:00
}
const itemDetails = this.getItem(tpl);
if (!itemDetails[0])
{
return false;
}
// Is item valid
2023-11-13 17:07:59 +01:00
return !itemDetails[1]._props.QuestItem &&
itemDetails[1]._type === "Item" &&
invalidBaseTypes.every((x) => !this.isOfBaseclass(tpl, x)) &&
this.getItemPrice(tpl) > 0 &&
!this.itemFilterService.isItemBlacklisted(tpl);
2023-03-03 16:23:46 +01:00
}
/**
* Check if the tpl / template Id provided is a descendent of the baseclass
*
* @param {string} tpl the item template id to check
* @param {string} baseClassTpl the baseclass to check for
* @return {boolean} is the tpl a descendent?
*/
public isOfBaseclass(tpl: string, baseClassTpl: string): boolean
{
return this.itemBaseClassService.itemHasBaseClass(tpl, [baseClassTpl]);
}
/**
* Check if item has any of the supplied base classes
* @param tpl Item to check base classes of
* @param baseClassTpls base classes to check for
* @returns true if any supplied base classes match
*/
public isOfBaseclasses(tpl: string, baseClassTpls: string[]): boolean
{
return this.itemBaseClassService.itemHasBaseClass(tpl, baseClassTpls);
}
/**
* Returns the item price based on the handbook or as a fallback from the prices.json if the item is not
* found in the handbook. If the price can't be found at all return 0
* @param tpl Item to look price up of
* @returns Price in roubles
2023-03-03 16:23:46 +01:00
*/
public getItemPrice(tpl: string): number
{
const handbookPrice = this.getStaticItemPrice(tpl);
if (handbookPrice >= 1)
{
return handbookPrice;
}
const dynamicPrice = this.getDynamicItemPrice(tpl);
if (dynamicPrice)
{
return dynamicPrice;
}
return 0;
}
/**
* Returns the item price based on the handbook or as a fallback from the prices.json if the item is not
* found in the handbook. If the price can't be found at all return 0
* @param tpl Item to look price up of
* @returns Price in roubles
*/
public getItemMaxPrice(tpl: string): number
{
const staticPrice = this.getStaticItemPrice(tpl);
const dynamicPrice = this.getDynamicItemPrice(tpl);
Refactor of InsuranceController & New ItemHelper Methods (!151) This commit is my second go-around at refactoring the `InsuranceController`, attempting to improving the code's modularity, maintainability, and efficiency while squashing a few bugs along the way. 1. **InsuranceController.ts** - Removed `ITemplateItem` import, as it is no longer used. - Introduced the `adoptOrphanedItems` method to manage orphaned items in the insurance list. - Since "normal" items are individually rolled for deletion, and can be nested within one another, there are situations where a parent item is deleted, leaving its children orphaned. This method moves those orphaned children from their missing parent into the root of the insurance container. - Overhauled `findItemsToDelete` method to improve its efficiency and readability: - Divided the original monolithic method into smaller, specialized methods like `populateItemsMap`, `populateParentAttachmentsMap`, `processRegularItems`, and `processAttachments`. - Changed the return type to `Set<string>` for better performance. - Introduced `EnrichedItem` interface (a simple extension of the `Item` interface) to add additional item data, like `name` and `maxPrice` to `Item` objects as they're processed throughout the class. This is done in place of repeatedly querying for this data, or complicating return types. - Enhanced logging capabilities to debug the item deletion process. Due to the *current* lack of testing available I've stepped up the amount of debug logging that is done. This will hopefully help us find issues in the future. - Modified the `rollForItemDelete` method, now renamed to `rollForDelete`, to include more detailed logging, return a boolean directly, and changed the `insuredItem` parameter to be optional. - Added new methods for dealing with some of the particulars that arise from item adoption and creating item maps. - Improved inline comments and documentation for better code maintainability. 2. **ItemHelper.ts** - Added the `isRaidModdable` method to check if an item is *actually* modifiable in-raid, which takes into account not just the item, but the item that it's attached to. - Added the `getAttachmentMainParent` method to fetch the main parent item of a given attachment, useful for item hierarchy traversal. For example, if you pass it an item ID of a suppressor, it will traverse up the muzzle brake, barrel, upper receiver, and return the gun that the suppressor is ultimately attached to, even if that gun is located within other multiple containers. - Added the `isAttachmentAttached` method to check if an item is an attachment that is currently attached to its parent. **Fixes:** - Resolved an issue that caused item attachments from being property grouped together for deletion rolls. This issue prevented valuable attachments from being taken first. - Resolved an issue that caused child items being orphaned when their parent was removed due to an insurance roll. Probable cause of the bug that made the client spaz out and send repeated insurance packages to the profile---Though I'm still unable to reproduce. - Probably more... Co-authored-by: Refringe <brownelltyler@gmail.com> Reviewed-on: https://dev.sp-tarkov.com/SPT-AKI/Server/pulls/151 Co-authored-by: Refringe <refringe@noreply.dev.sp-tarkov.com> Co-committed-by: Refringe <refringe@noreply.dev.sp-tarkov.com>
2023-10-14 11:05:49 +02:00
return Math.max(staticPrice, dynamicPrice);
}
/**
* Get the static (handbook) price in roubles for an item by tpl
* @param tpl Items tpl id to look up price
* @returns Price in roubles (0 if not found)
*/
public getStaticItemPrice(tpl: string): number
2023-03-03 16:23:46 +01:00
{
const handbookPrice = this.handbookHelper.getTemplatePrice(tpl);
if (handbookPrice >= 1)
2023-03-03 16:23:46 +01:00
{
return handbookPrice;
}
return 0;
}
/**
* Get the dynamic (flea) price in roubles for an item by tpl
* @param tpl Items tpl id to look up price
* @returns Price in roubles (undefined if not found)
*/
public getDynamicItemPrice(tpl: string): number
{
2023-03-03 16:23:46 +01:00
const dynamicPrice = this.databaseServer.getTables().templates.prices[tpl];
if (dynamicPrice)
{
return dynamicPrice;
}
return 0;
Refactor of InsuranceController & New ItemHelper Methods (!151) This commit is my second go-around at refactoring the `InsuranceController`, attempting to improving the code's modularity, maintainability, and efficiency while squashing a few bugs along the way. 1. **InsuranceController.ts** - Removed `ITemplateItem` import, as it is no longer used. - Introduced the `adoptOrphanedItems` method to manage orphaned items in the insurance list. - Since "normal" items are individually rolled for deletion, and can be nested within one another, there are situations where a parent item is deleted, leaving its children orphaned. This method moves those orphaned children from their missing parent into the root of the insurance container. - Overhauled `findItemsToDelete` method to improve its efficiency and readability: - Divided the original monolithic method into smaller, specialized methods like `populateItemsMap`, `populateParentAttachmentsMap`, `processRegularItems`, and `processAttachments`. - Changed the return type to `Set<string>` for better performance. - Introduced `EnrichedItem` interface (a simple extension of the `Item` interface) to add additional item data, like `name` and `maxPrice` to `Item` objects as they're processed throughout the class. This is done in place of repeatedly querying for this data, or complicating return types. - Enhanced logging capabilities to debug the item deletion process. Due to the *current* lack of testing available I've stepped up the amount of debug logging that is done. This will hopefully help us find issues in the future. - Modified the `rollForItemDelete` method, now renamed to `rollForDelete`, to include more detailed logging, return a boolean directly, and changed the `insuredItem` parameter to be optional. - Added new methods for dealing with some of the particulars that arise from item adoption and creating item maps. - Improved inline comments and documentation for better code maintainability. 2. **ItemHelper.ts** - Added the `isRaidModdable` method to check if an item is *actually* modifiable in-raid, which takes into account not just the item, but the item that it's attached to. - Added the `getAttachmentMainParent` method to fetch the main parent item of a given attachment, useful for item hierarchy traversal. For example, if you pass it an item ID of a suppressor, it will traverse up the muzzle brake, barrel, upper receiver, and return the gun that the suppressor is ultimately attached to, even if that gun is located within other multiple containers. - Added the `isAttachmentAttached` method to check if an item is an attachment that is currently attached to its parent. **Fixes:** - Resolved an issue that caused item attachments from being property grouped together for deletion rolls. This issue prevented valuable attachments from being taken first. - Resolved an issue that caused child items being orphaned when their parent was removed due to an insurance roll. Probable cause of the bug that made the client spaz out and send repeated insurance packages to the profile---Though I'm still unable to reproduce. - Probably more... Co-authored-by: Refringe <brownelltyler@gmail.com> Reviewed-on: https://dev.sp-tarkov.com/SPT-AKI/Server/pulls/151 Co-authored-by: Refringe <refringe@noreply.dev.sp-tarkov.com> Co-committed-by: Refringe <refringe@noreply.dev.sp-tarkov.com>
2023-10-14 11:05:49 +02:00
}
2023-03-03 16:23:46 +01:00
/**
* Update items upd.StackObjectsCount to be 1 if its upd is missing or StackObjectsCount is undefined
* @param item Item to update
* @returns Fixed item
*/
2023-03-03 16:23:46 +01:00
public fixItemStackCount(item: Item): Item
{
if (item.upd === undefined)
{
item.upd = {
2023-11-13 17:07:59 +01:00
StackObjectsCount: 1,
2023-03-03 16:23:46 +01:00
};
}
if (item.upd.StackObjectsCount === undefined)
{
item.upd.StackObjectsCount = 1;
}
return item;
}
/**
* AmmoBoxes contain StackSlots which need to be filled for the AmmoBox to have content.
* Here's what a filled AmmoBox looks like:
* {
2023-10-31 23:52:09 +01:00
* _id: "b1bbe982daa00ac841d4ae4d",
* _tpl: "57372c89245977685d4159b1",
* parentId: "5fe49a0e2694b0755a504876",
* slotId: "hideout",
* location: {
* x: 3,
* y: 4,
* r: 0
2023-03-03 16:23:46 +01:00
* },
2023-10-31 23:52:09 +01:00
* upd: {
* StackObjectsCount: 1
2023-03-03 16:23:46 +01:00
* }
* },
* {
2023-10-31 23:52:09 +01:00
* _id: "b997b4117199033afd274a06",
* _tpl: "56dff061d2720bb5668b4567",
* parentId: "b1bbe982daa00ac841d4ae4d",
* slotId: "cartridges",
* location: 0,
* upd: {
* StackObjectsCount: 30
2023-03-03 16:23:46 +01:00
* }
* }
* Given the AmmoBox Item (first object) this function generates the StackSlot (second object) and returns it.
* StackSlots are only used for AmmoBoxes which only have one element in StackSlots. However, it seems to be generic
* to possibly also have more than one StackSlot. As good as possible, without seeing items having more than one
* StackSlot, this function takes account of this and creates and returns an array of StackSlotItems
*
* @param {object} item The item template of the AmmoBox as given in items.json
* @param {string} parentId The id of the AmmoBox instance these StackSlotItems should be children of
* @returns {array} The array of StackSlotItems
*/
public generateItemsFromStackSlot(item: ITemplateItem, parentId: string): Item[]
{
const stackSlotItems: Item[] = [];
// This is a AmmoBox or something other with Stackslots (nothing exists yet besides AmmoBoxes afaik)
for (const stackSlot of item._props.StackSlots)
{
const slotId = stackSlot._name;
const count = stackSlot._max_count;
// those are all arrays. For AmmoBoxes it's only one element each so we take 0 hardcoded
// not sure if at any point there will be more than one element - but what so take then?
const ammoTpl = stackSlot._props.filters[0].Filter[0];
if (ammoTpl)
{
const stackSlotItem: Item = {
_id: this.hashUtil.generate(),
_tpl: ammoTpl,
parentId: parentId,
slotId: slotId,
location: 0,
upd: {
2023-11-13 17:07:59 +01:00
StackObjectsCount: count,
},
2023-03-03 16:23:46 +01:00
};
stackSlotItems.push(stackSlotItem);
}
else
{
this.logger.warning(`No ids found in Filter for StackSlot ${slotId} of Item ${item._id}.`);
}
}
return stackSlotItems;
}
/**
* Get cloned copy of all item data from items.json
* @returns array of ITemplateItem objects
*/
public getItems(): ITemplateItem[]
{
return this.jsonUtil.clone(Object.values(this.databaseServer.getTables().templates.items));
}
/**
* Gets item data from items.json
* @param tpl items template id to look up
* @returns bool - is valid + template item object as array
*/
public getItem(tpl: string): [boolean, ITemplateItem]
{
// -> Gets item from <input: _tpl>
if (tpl in this.databaseServer.getTables().templates.items)
{
return [true, this.databaseServer.getTables().templates.items[tpl]];
}
return [false, undefined];
}
public isItemInDb(tpl: string): boolean
{
const itemDetails = this.getItem(tpl);
return itemDetails[0];
}
2023-03-03 16:23:46 +01:00
/**
* get normalized value (0-1) based on item condition
Refactor of InsuranceController & New ItemHelper Methods (!151) This commit is my second go-around at refactoring the `InsuranceController`, attempting to improving the code's modularity, maintainability, and efficiency while squashing a few bugs along the way. 1. **InsuranceController.ts** - Removed `ITemplateItem` import, as it is no longer used. - Introduced the `adoptOrphanedItems` method to manage orphaned items in the insurance list. - Since "normal" items are individually rolled for deletion, and can be nested within one another, there are situations where a parent item is deleted, leaving its children orphaned. This method moves those orphaned children from their missing parent into the root of the insurance container. - Overhauled `findItemsToDelete` method to improve its efficiency and readability: - Divided the original monolithic method into smaller, specialized methods like `populateItemsMap`, `populateParentAttachmentsMap`, `processRegularItems`, and `processAttachments`. - Changed the return type to `Set<string>` for better performance. - Introduced `EnrichedItem` interface (a simple extension of the `Item` interface) to add additional item data, like `name` and `maxPrice` to `Item` objects as they're processed throughout the class. This is done in place of repeatedly querying for this data, or complicating return types. - Enhanced logging capabilities to debug the item deletion process. Due to the *current* lack of testing available I've stepped up the amount of debug logging that is done. This will hopefully help us find issues in the future. - Modified the `rollForItemDelete` method, now renamed to `rollForDelete`, to include more detailed logging, return a boolean directly, and changed the `insuredItem` parameter to be optional. - Added new methods for dealing with some of the particulars that arise from item adoption and creating item maps. - Improved inline comments and documentation for better code maintainability. 2. **ItemHelper.ts** - Added the `isRaidModdable` method to check if an item is *actually* modifiable in-raid, which takes into account not just the item, but the item that it's attached to. - Added the `getAttachmentMainParent` method to fetch the main parent item of a given attachment, useful for item hierarchy traversal. For example, if you pass it an item ID of a suppressor, it will traverse up the muzzle brake, barrel, upper receiver, and return the gun that the suppressor is ultimately attached to, even if that gun is located within other multiple containers. - Added the `isAttachmentAttached` method to check if an item is an attachment that is currently attached to its parent. **Fixes:** - Resolved an issue that caused item attachments from being property grouped together for deletion rolls. This issue prevented valuable attachments from being taken first. - Resolved an issue that caused child items being orphaned when their parent was removed due to an insurance roll. Probable cause of the bug that made the client spaz out and send repeated insurance packages to the profile---Though I'm still unable to reproduce. - Probably more... Co-authored-by: Refringe <brownelltyler@gmail.com> Reviewed-on: https://dev.sp-tarkov.com/SPT-AKI/Server/pulls/151 Co-authored-by: Refringe <refringe@noreply.dev.sp-tarkov.com> Co-committed-by: Refringe <refringe@noreply.dev.sp-tarkov.com>
2023-10-14 11:05:49 +02:00
* @param item
2023-03-03 16:23:46 +01:00
* @returns number between 0 and 1
*/
public getItemQualityModifier(item: Item): number
{
// Default to 100%
let result = 1;
if (item.upd)
{
const medkit = (item.upd.MedKit) ? item.upd.MedKit : null;
const repairable = (item.upd.Repairable) ? item.upd.Repairable : null;
const foodDrink = (item.upd.FoodDrink) ? item.upd.FoodDrink : null;
const key = (item.upd.Key) ? item.upd.Key : null;
const resource = (item.upd.Resource) ? item.upd.Resource : null;
const repairKit = (item.upd.RepairKit) ? item.upd.RepairKit : null;
const itemDetails = this.getItem(item._tpl)[1];
if (medkit)
{
// Meds
result = medkit.HpResource / itemDetails._props.MaxHpResource;
}
else if (repairable)
{
result = this.getRepairableItemQualityValue(itemDetails, repairable, item);
}
else if (foodDrink)
{
// food & drink
result = foodDrink.HpPercent / itemDetails._props.MaxResource;
}
else if (key && key.NumberOfUsages > 0)
{
// keys - keys count upwards, not down like everything else
const maxNumOfUsages = itemDetails._props.MaximumNumberOfUsage;
result = (maxNumOfUsages - key.NumberOfUsages) / maxNumOfUsages;
}
else if (resource && resource.UnitsConsumed > 0)
{
// Things like fuel tank
result = resource.Value / itemDetails._props.MaxResource;
2023-03-03 16:23:46 +01:00
}
else if (repairKit)
{
// Repair kits
result = repairKit.Resource / itemDetails._props.MaxRepairResource;
}
if (result === 0)
{
// make item non-zero but still very low
result = 0.01;
}
}
return result;
}
/**
* Get a quality value based on a repairable items (weapon/armor) current state between current and max durability
* @param itemDetails Db details for item we want quality value for
* @param repairable Repairable properties
* @param item Item quality value is for
* @returns A number between 0 and 1
2023-03-03 16:23:46 +01:00
*/
protected getRepairableItemQualityValue(itemDetails: ITemplateItem, repairable: Repairable, item: Item): number
{
// Edge case, max durability is below durability
2023-11-06 10:24:13 +01:00
if (repairable.Durability > repairable.MaxDurability)
{
2023-11-13 17:07:59 +01:00
this.logger.warning(
`Max durability: ${repairable.MaxDurability} for item id: ${item._id} was below Durability: ${repairable.Durability}, adjusting values to match`,
);
repairable.MaxDurability = repairable.Durability;
}
2023-03-03 16:23:46 +01:00
// Armor
if (itemDetails._props.armorClass)
{
return repairable.Durability / itemDetails._props.MaxDurability;
}
// Weapon
// Get max dura from props, if it isnt there use repairable max dura value
2023-11-13 17:07:59 +01:00
const maxDurability = (itemDetails._props.MaxDurability) ?
itemDetails._props.MaxDurability :
repairable.MaxDurability;
const durability = repairable.Durability / maxDurability;
2023-03-03 16:23:46 +01:00
if (!durability)
{
this.logger.error(this.localisationService.getText("item-durability_value_invalid_use_default", item._tpl));
2023-03-03 16:23:46 +01:00
return 1;
2023-03-03 16:23:46 +01:00
}
return Math.sqrt(durability);
2023-03-03 16:23:46 +01:00
}
/**
2023-04-24 13:47:19 +02:00
* Recursive function that looks at every item from parameter and gets their childrens Ids + includes parent item in results
* @param items Array of items (item + possible children)
* @param itemId Parent items id
2023-03-03 16:23:46 +01:00
* @returns an array of strings
*/
public findAndReturnChildrenByItems(items: Item[], itemId: string): string[]
2023-03-03 16:23:46 +01:00
{
const list: string[] = [];
for (const childitem of items)
{
if (childitem.parentId === itemId)
2023-03-03 16:23:46 +01:00
{
list.push(...this.findAndReturnChildrenByItems(items, childitem._id));
}
}
list.push(itemId); // Required, push original item id onto array
2023-03-03 16:23:46 +01:00
return list;
}
/**
* A variant of findAndReturnChildren where the output is list of item objects instead of their ids.
Refactor of InsuranceController & New ItemHelper Methods (!151) This commit is my second go-around at refactoring the `InsuranceController`, attempting to improving the code's modularity, maintainability, and efficiency while squashing a few bugs along the way. 1. **InsuranceController.ts** - Removed `ITemplateItem` import, as it is no longer used. - Introduced the `adoptOrphanedItems` method to manage orphaned items in the insurance list. - Since "normal" items are individually rolled for deletion, and can be nested within one another, there are situations where a parent item is deleted, leaving its children orphaned. This method moves those orphaned children from their missing parent into the root of the insurance container. - Overhauled `findItemsToDelete` method to improve its efficiency and readability: - Divided the original monolithic method into smaller, specialized methods like `populateItemsMap`, `populateParentAttachmentsMap`, `processRegularItems`, and `processAttachments`. - Changed the return type to `Set<string>` for better performance. - Introduced `EnrichedItem` interface (a simple extension of the `Item` interface) to add additional item data, like `name` and `maxPrice` to `Item` objects as they're processed throughout the class. This is done in place of repeatedly querying for this data, or complicating return types. - Enhanced logging capabilities to debug the item deletion process. Due to the *current* lack of testing available I've stepped up the amount of debug logging that is done. This will hopefully help us find issues in the future. - Modified the `rollForItemDelete` method, now renamed to `rollForDelete`, to include more detailed logging, return a boolean directly, and changed the `insuredItem` parameter to be optional. - Added new methods for dealing with some of the particulars that arise from item adoption and creating item maps. - Improved inline comments and documentation for better code maintainability. 2. **ItemHelper.ts** - Added the `isRaidModdable` method to check if an item is *actually* modifiable in-raid, which takes into account not just the item, but the item that it's attached to. - Added the `getAttachmentMainParent` method to fetch the main parent item of a given attachment, useful for item hierarchy traversal. For example, if you pass it an item ID of a suppressor, it will traverse up the muzzle brake, barrel, upper receiver, and return the gun that the suppressor is ultimately attached to, even if that gun is located within other multiple containers. - Added the `isAttachmentAttached` method to check if an item is an attachment that is currently attached to its parent. **Fixes:** - Resolved an issue that caused item attachments from being property grouped together for deletion rolls. This issue prevented valuable attachments from being taken first. - Resolved an issue that caused child items being orphaned when their parent was removed due to an insurance roll. Probable cause of the bug that made the client spaz out and send repeated insurance packages to the profile---Though I'm still unable to reproduce. - Probably more... Co-authored-by: Refringe <brownelltyler@gmail.com> Reviewed-on: https://dev.sp-tarkov.com/SPT-AKI/Server/pulls/151 Co-authored-by: Refringe <refringe@noreply.dev.sp-tarkov.com> Co-committed-by: Refringe <refringe@noreply.dev.sp-tarkov.com>
2023-10-14 11:05:49 +02:00
* @param items
* @param baseItemId
* @returns An array of Item objects
2023-03-03 16:23:46 +01:00
*/
public findAndReturnChildrenAsItems(items: Item[], baseItemId: string): Item[]
{
const list: Item[] = [];
for (const childItem of items)
{
// Include itself
2023-03-03 16:23:46 +01:00
if (childItem._id === baseItemId)
{
list.unshift(childItem);
continue;
}
2023-11-13 17:07:59 +01:00
if (childItem.parentId === baseItemId && !list.find((item) => childItem._id === item._id))
2023-03-03 16:23:46 +01:00
{
list.push(...this.findAndReturnChildrenAsItems(items, childItem._id));
}
}
2023-03-03 16:23:46 +01:00
return list;
}
/**
* Find children of the item in a given assort (weapons parts for example, need recursive loop function)
* @param itemIdToFind Template id of item to check for
* @param assort Array of items to check in
* @returns Array of children of requested item
*/
public findAndReturnChildrenByAssort(itemIdToFind: string, assort: Item[]): Item[]
{
let list: Item[] = [];
for (const itemFromAssort of assort)
{
2023-11-13 17:07:59 +01:00
if (itemFromAssort.parentId === itemIdToFind && !list.find((item) => itemFromAssort._id === item._id))
2023-03-03 16:23:46 +01:00
{
list.push(itemFromAssort);
list = list.concat(this.findAndReturnChildrenByAssort(itemFromAssort._id, assort));
}
}
return list;
}
/**
* Check if the passed in item has buy count restrictions
* @param itemToCheck Item to check
* @returns true if it has buy restrictions
*/
public hasBuyRestrictions(itemToCheck: Item): boolean
{
2023-11-13 17:07:59 +01:00
if (
itemToCheck.upd?.BuyRestrictionCurrent !== undefined &&
itemToCheck.upd?.BuyRestrictionMax !== undefined
)
2023-03-03 16:23:46 +01:00
{
return true;
}
return false;
}
/**
* is the passed in template id a dog tag
* @param tpl Template id to check
* @returns true if it is a dogtag
*/
public isDogtag(tpl: string): boolean
{
return tpl === BaseClasses.DOG_TAG_BEAR || tpl === BaseClasses.DOG_TAG_USEC;
}
/**
* Gets the identifier for a child using slotId, locationX and locationY.
Refactor of InsuranceController & New ItemHelper Methods (!151) This commit is my second go-around at refactoring the `InsuranceController`, attempting to improving the code's modularity, maintainability, and efficiency while squashing a few bugs along the way. 1. **InsuranceController.ts** - Removed `ITemplateItem` import, as it is no longer used. - Introduced the `adoptOrphanedItems` method to manage orphaned items in the insurance list. - Since "normal" items are individually rolled for deletion, and can be nested within one another, there are situations where a parent item is deleted, leaving its children orphaned. This method moves those orphaned children from their missing parent into the root of the insurance container. - Overhauled `findItemsToDelete` method to improve its efficiency and readability: - Divided the original monolithic method into smaller, specialized methods like `populateItemsMap`, `populateParentAttachmentsMap`, `processRegularItems`, and `processAttachments`. - Changed the return type to `Set<string>` for better performance. - Introduced `EnrichedItem` interface (a simple extension of the `Item` interface) to add additional item data, like `name` and `maxPrice` to `Item` objects as they're processed throughout the class. This is done in place of repeatedly querying for this data, or complicating return types. - Enhanced logging capabilities to debug the item deletion process. Due to the *current* lack of testing available I've stepped up the amount of debug logging that is done. This will hopefully help us find issues in the future. - Modified the `rollForItemDelete` method, now renamed to `rollForDelete`, to include more detailed logging, return a boolean directly, and changed the `insuredItem` parameter to be optional. - Added new methods for dealing with some of the particulars that arise from item adoption and creating item maps. - Improved inline comments and documentation for better code maintainability. 2. **ItemHelper.ts** - Added the `isRaidModdable` method to check if an item is *actually* modifiable in-raid, which takes into account not just the item, but the item that it's attached to. - Added the `getAttachmentMainParent` method to fetch the main parent item of a given attachment, useful for item hierarchy traversal. For example, if you pass it an item ID of a suppressor, it will traverse up the muzzle brake, barrel, upper receiver, and return the gun that the suppressor is ultimately attached to, even if that gun is located within other multiple containers. - Added the `isAttachmentAttached` method to check if an item is an attachment that is currently attached to its parent. **Fixes:** - Resolved an issue that caused item attachments from being property grouped together for deletion rolls. This issue prevented valuable attachments from being taken first. - Resolved an issue that caused child items being orphaned when their parent was removed due to an insurance roll. Probable cause of the bug that made the client spaz out and send repeated insurance packages to the profile---Though I'm still unable to reproduce. - Probably more... Co-authored-by: Refringe <brownelltyler@gmail.com> Reviewed-on: https://dev.sp-tarkov.com/SPT-AKI/Server/pulls/151 Co-authored-by: Refringe <refringe@noreply.dev.sp-tarkov.com> Co-committed-by: Refringe <refringe@noreply.dev.sp-tarkov.com>
2023-10-14 11:05:49 +02:00
* @param item
2023-03-03 16:23:46 +01:00
* @returns "slotId OR slotid,locationX,locationY"
*/
public getChildId(item: Item): string
{
if (!("location" in item))
{
return item.slotId;
}
return `${item.slotId},${(item.location as Location).x},${(item.location as Location).y}`;
}
/**
* Can the passed in item be stacked
* @param tpl item to check
* @returns true if it can be stacked
*/
public isItemTplStackable(tpl: string): boolean
{
const item = this.databaseServer.getTables().templates.items[tpl];
if (!item)
{
return undefined;
}
return item._props.StackMaxSize > 1;
2023-03-03 16:23:46 +01:00
}
/**
* split item stack if it exceeds its items StackMaxSize property
* @param itemToSplit Item to split into smaller stacks
* @returns Array of split items
2023-03-03 16:23:46 +01:00
*/
public splitStack(itemToSplit: Item): Item[]
2023-03-03 16:23:46 +01:00
{
if (!(itemToSplit?.upd?.StackObjectsCount != null))
2023-03-03 16:23:46 +01:00
{
return [itemToSplit];
2023-03-03 16:23:46 +01:00
}
const maxStackSize = this.databaseServer.getTables().templates.items[itemToSplit._tpl]._props.StackMaxSize;
let remainingCount = itemToSplit.upd.StackObjectsCount;
2023-03-03 16:23:46 +01:00
const stacks: Item[] = [];
// If the current count is already equal or less than the max
// then just return the item as is.
if (remainingCount <= maxStackSize)
2023-03-03 16:23:46 +01:00
{
stacks.push(this.jsonUtil.clone(itemToSplit));
2023-03-03 16:23:46 +01:00
return stacks;
}
while (remainingCount)
2023-03-03 16:23:46 +01:00
{
const amount = Math.min(remainingCount, maxStackSize);
const newStack = this.jsonUtil.clone(itemToSplit);
2023-03-03 16:23:46 +01:00
newStack._id = this.hashUtil.generate();
newStack.upd.StackObjectsCount = amount;
remainingCount -= amount;
2023-03-03 16:23:46 +01:00
stacks.push(newStack);
}
return stacks;
}
/**
* Find Barter items from array of items
* @param {string} by tpl or id
* @param {Item[]} items Array of items to iterate over
2023-03-03 16:23:46 +01:00
* @param {string} barterItemId
* @returns Array of Item objects
*/
public findBarterItems(by: "tpl" | "id", items: Item[], barterItemId: string): Item[]
Refactor of InsuranceController & New ItemHelper Methods (!151) This commit is my second go-around at refactoring the `InsuranceController`, attempting to improving the code's modularity, maintainability, and efficiency while squashing a few bugs along the way. 1. **InsuranceController.ts** - Removed `ITemplateItem` import, as it is no longer used. - Introduced the `adoptOrphanedItems` method to manage orphaned items in the insurance list. - Since "normal" items are individually rolled for deletion, and can be nested within one another, there are situations where a parent item is deleted, leaving its children orphaned. This method moves those orphaned children from their missing parent into the root of the insurance container. - Overhauled `findItemsToDelete` method to improve its efficiency and readability: - Divided the original monolithic method into smaller, specialized methods like `populateItemsMap`, `populateParentAttachmentsMap`, `processRegularItems`, and `processAttachments`. - Changed the return type to `Set<string>` for better performance. - Introduced `EnrichedItem` interface (a simple extension of the `Item` interface) to add additional item data, like `name` and `maxPrice` to `Item` objects as they're processed throughout the class. This is done in place of repeatedly querying for this data, or complicating return types. - Enhanced logging capabilities to debug the item deletion process. Due to the *current* lack of testing available I've stepped up the amount of debug logging that is done. This will hopefully help us find issues in the future. - Modified the `rollForItemDelete` method, now renamed to `rollForDelete`, to include more detailed logging, return a boolean directly, and changed the `insuredItem` parameter to be optional. - Added new methods for dealing with some of the particulars that arise from item adoption and creating item maps. - Improved inline comments and documentation for better code maintainability. 2. **ItemHelper.ts** - Added the `isRaidModdable` method to check if an item is *actually* modifiable in-raid, which takes into account not just the item, but the item that it's attached to. - Added the `getAttachmentMainParent` method to fetch the main parent item of a given attachment, useful for item hierarchy traversal. For example, if you pass it an item ID of a suppressor, it will traverse up the muzzle brake, barrel, upper receiver, and return the gun that the suppressor is ultimately attached to, even if that gun is located within other multiple containers. - Added the `isAttachmentAttached` method to check if an item is an attachment that is currently attached to its parent. **Fixes:** - Resolved an issue that caused item attachments from being property grouped together for deletion rolls. This issue prevented valuable attachments from being taken first. - Resolved an issue that caused child items being orphaned when their parent was removed due to an insurance roll. Probable cause of the bug that made the client spaz out and send repeated insurance packages to the profile---Though I'm still unable to reproduce. - Probably more... Co-authored-by: Refringe <brownelltyler@gmail.com> Reviewed-on: https://dev.sp-tarkov.com/SPT-AKI/Server/pulls/151 Co-authored-by: Refringe <refringe@noreply.dev.sp-tarkov.com> Co-committed-by: Refringe <refringe@noreply.dev.sp-tarkov.com>
2023-10-14 11:05:49 +02:00
{
2023-03-03 16:23:46 +01:00
// find required items to take after buying (handles multiple items)
2023-11-13 17:07:59 +01:00
const barterIDs = typeof barterItemId === "string" ?
[barterItemId] :
barterItemId;
2023-03-03 16:23:46 +01:00
let barterItems: Item[] = [];
2023-03-03 16:23:46 +01:00
for (const barterID of barterIDs)
{
2023-11-13 17:07:59 +01:00
const filterResult = items.filter((item) =>
2023-03-03 16:23:46 +01:00
{
2023-11-13 17:07:59 +01:00
return by === "tpl" ?
(item._tpl === barterID) :
(item._id === barterID);
2023-03-03 16:23:46 +01:00
});
barterItems = Object.assign(barterItems, filterResult);
}
Refactor of InsuranceController & New ItemHelper Methods (!151) This commit is my second go-around at refactoring the `InsuranceController`, attempting to improving the code's modularity, maintainability, and efficiency while squashing a few bugs along the way. 1. **InsuranceController.ts** - Removed `ITemplateItem` import, as it is no longer used. - Introduced the `adoptOrphanedItems` method to manage orphaned items in the insurance list. - Since "normal" items are individually rolled for deletion, and can be nested within one another, there are situations where a parent item is deleted, leaving its children orphaned. This method moves those orphaned children from their missing parent into the root of the insurance container. - Overhauled `findItemsToDelete` method to improve its efficiency and readability: - Divided the original monolithic method into smaller, specialized methods like `populateItemsMap`, `populateParentAttachmentsMap`, `processRegularItems`, and `processAttachments`. - Changed the return type to `Set<string>` for better performance. - Introduced `EnrichedItem` interface (a simple extension of the `Item` interface) to add additional item data, like `name` and `maxPrice` to `Item` objects as they're processed throughout the class. This is done in place of repeatedly querying for this data, or complicating return types. - Enhanced logging capabilities to debug the item deletion process. Due to the *current* lack of testing available I've stepped up the amount of debug logging that is done. This will hopefully help us find issues in the future. - Modified the `rollForItemDelete` method, now renamed to `rollForDelete`, to include more detailed logging, return a boolean directly, and changed the `insuredItem` parameter to be optional. - Added new methods for dealing with some of the particulars that arise from item adoption and creating item maps. - Improved inline comments and documentation for better code maintainability. 2. **ItemHelper.ts** - Added the `isRaidModdable` method to check if an item is *actually* modifiable in-raid, which takes into account not just the item, but the item that it's attached to. - Added the `getAttachmentMainParent` method to fetch the main parent item of a given attachment, useful for item hierarchy traversal. For example, if you pass it an item ID of a suppressor, it will traverse up the muzzle brake, barrel, upper receiver, and return the gun that the suppressor is ultimately attached to, even if that gun is located within other multiple containers. - Added the `isAttachmentAttached` method to check if an item is an attachment that is currently attached to its parent. **Fixes:** - Resolved an issue that caused item attachments from being property grouped together for deletion rolls. This issue prevented valuable attachments from being taken first. - Resolved an issue that caused child items being orphaned when their parent was removed due to an insurance roll. Probable cause of the bug that made the client spaz out and send repeated insurance packages to the profile---Though I'm still unable to reproduce. - Probably more... Co-authored-by: Refringe <brownelltyler@gmail.com> Reviewed-on: https://dev.sp-tarkov.com/SPT-AKI/Server/pulls/151 Co-authored-by: Refringe <refringe@noreply.dev.sp-tarkov.com> Co-committed-by: Refringe <refringe@noreply.dev.sp-tarkov.com>
2023-10-14 11:05:49 +02:00
if (barterItems.length === 0)
{
this.logger.warning(`No items found for barter Id: ${barterIDs}`);
2023-03-03 16:23:46 +01:00
}
Refactor of InsuranceController & New ItemHelper Methods (!151) This commit is my second go-around at refactoring the `InsuranceController`, attempting to improving the code's modularity, maintainability, and efficiency while squashing a few bugs along the way. 1. **InsuranceController.ts** - Removed `ITemplateItem` import, as it is no longer used. - Introduced the `adoptOrphanedItems` method to manage orphaned items in the insurance list. - Since "normal" items are individually rolled for deletion, and can be nested within one another, there are situations where a parent item is deleted, leaving its children orphaned. This method moves those orphaned children from their missing parent into the root of the insurance container. - Overhauled `findItemsToDelete` method to improve its efficiency and readability: - Divided the original monolithic method into smaller, specialized methods like `populateItemsMap`, `populateParentAttachmentsMap`, `processRegularItems`, and `processAttachments`. - Changed the return type to `Set<string>` for better performance. - Introduced `EnrichedItem` interface (a simple extension of the `Item` interface) to add additional item data, like `name` and `maxPrice` to `Item` objects as they're processed throughout the class. This is done in place of repeatedly querying for this data, or complicating return types. - Enhanced logging capabilities to debug the item deletion process. Due to the *current* lack of testing available I've stepped up the amount of debug logging that is done. This will hopefully help us find issues in the future. - Modified the `rollForItemDelete` method, now renamed to `rollForDelete`, to include more detailed logging, return a boolean directly, and changed the `insuredItem` parameter to be optional. - Added new methods for dealing with some of the particulars that arise from item adoption and creating item maps. - Improved inline comments and documentation for better code maintainability. 2. **ItemHelper.ts** - Added the `isRaidModdable` method to check if an item is *actually* modifiable in-raid, which takes into account not just the item, but the item that it's attached to. - Added the `getAttachmentMainParent` method to fetch the main parent item of a given attachment, useful for item hierarchy traversal. For example, if you pass it an item ID of a suppressor, it will traverse up the muzzle brake, barrel, upper receiver, and return the gun that the suppressor is ultimately attached to, even if that gun is located within other multiple containers. - Added the `isAttachmentAttached` method to check if an item is an attachment that is currently attached to its parent. **Fixes:** - Resolved an issue that caused item attachments from being property grouped together for deletion rolls. This issue prevented valuable attachments from being taken first. - Resolved an issue that caused child items being orphaned when their parent was removed due to an insurance roll. Probable cause of the bug that made the client spaz out and send repeated insurance packages to the profile---Though I'm still unable to reproduce. - Probably more... Co-authored-by: Refringe <brownelltyler@gmail.com> Reviewed-on: https://dev.sp-tarkov.com/SPT-AKI/Server/pulls/151 Co-authored-by: Refringe <refringe@noreply.dev.sp-tarkov.com> Co-committed-by: Refringe <refringe@noreply.dev.sp-tarkov.com>
2023-10-14 11:05:49 +02:00
return barterItems;
2023-03-03 16:23:46 +01:00
}
/**
2023-07-13 15:20:31 +02:00
* Regenerate all guids with new ids, exceptions are for items that cannot be altered (e.g. stash/sorting table)
* @param pmcData Player profile
* @param items Items to adjust ID values of
2023-03-03 16:23:46 +01:00
* @param insuredItems insured items to not replace ids for
Refactor of InsuranceController & New ItemHelper Methods (!151) This commit is my second go-around at refactoring the `InsuranceController`, attempting to improving the code's modularity, maintainability, and efficiency while squashing a few bugs along the way. 1. **InsuranceController.ts** - Removed `ITemplateItem` import, as it is no longer used. - Introduced the `adoptOrphanedItems` method to manage orphaned items in the insurance list. - Since "normal" items are individually rolled for deletion, and can be nested within one another, there are situations where a parent item is deleted, leaving its children orphaned. This method moves those orphaned children from their missing parent into the root of the insurance container. - Overhauled `findItemsToDelete` method to improve its efficiency and readability: - Divided the original monolithic method into smaller, specialized methods like `populateItemsMap`, `populateParentAttachmentsMap`, `processRegularItems`, and `processAttachments`. - Changed the return type to `Set<string>` for better performance. - Introduced `EnrichedItem` interface (a simple extension of the `Item` interface) to add additional item data, like `name` and `maxPrice` to `Item` objects as they're processed throughout the class. This is done in place of repeatedly querying for this data, or complicating return types. - Enhanced logging capabilities to debug the item deletion process. Due to the *current* lack of testing available I've stepped up the amount of debug logging that is done. This will hopefully help us find issues in the future. - Modified the `rollForItemDelete` method, now renamed to `rollForDelete`, to include more detailed logging, return a boolean directly, and changed the `insuredItem` parameter to be optional. - Added new methods for dealing with some of the particulars that arise from item adoption and creating item maps. - Improved inline comments and documentation for better code maintainability. 2. **ItemHelper.ts** - Added the `isRaidModdable` method to check if an item is *actually* modifiable in-raid, which takes into account not just the item, but the item that it's attached to. - Added the `getAttachmentMainParent` method to fetch the main parent item of a given attachment, useful for item hierarchy traversal. For example, if you pass it an item ID of a suppressor, it will traverse up the muzzle brake, barrel, upper receiver, and return the gun that the suppressor is ultimately attached to, even if that gun is located within other multiple containers. - Added the `isAttachmentAttached` method to check if an item is an attachment that is currently attached to its parent. **Fixes:** - Resolved an issue that caused item attachments from being property grouped together for deletion rolls. This issue prevented valuable attachments from being taken first. - Resolved an issue that caused child items being orphaned when their parent was removed due to an insurance roll. Probable cause of the bug that made the client spaz out and send repeated insurance packages to the profile---Though I'm still unable to reproduce. - Probably more... Co-authored-by: Refringe <brownelltyler@gmail.com> Reviewed-on: https://dev.sp-tarkov.com/SPT-AKI/Server/pulls/151 Co-authored-by: Refringe <refringe@noreply.dev.sp-tarkov.com> Co-committed-by: Refringe <refringe@noreply.dev.sp-tarkov.com>
2023-10-14 11:05:49 +02:00
* @param fastPanel
2023-07-13 18:32:50 +02:00
* @returns Item[]
2023-03-03 16:23:46 +01:00
*/
2023-07-13 15:20:31 +02:00
public replaceIDs(pmcData: IPmcData, items: Item[], insuredItems: InsuredItem[] = null, fastPanel = null): Item[]
2023-03-03 16:23:46 +01:00
{
// replace bsg shit long ID with proper one
let serialisedInventory = this.jsonUtil.serialize(items);
for (const item of items)
{
if (pmcData !== null)
{
// Insured items shouldn't be renamed
// only works for pmcs.
2023-11-13 17:07:59 +01:00
if (insuredItems?.find((insuredItem) => insuredItem.itemId === item._id))
2023-03-03 16:23:46 +01:00
{
continue;
}
// Do not replace important ID's
2023-11-13 17:07:59 +01:00
if (
item._id === pmcData.Inventory.equipment ||
item._id === pmcData.Inventory.questRaidItems ||
item._id === pmcData.Inventory.questStashItems ||
item._id === pmcData.Inventory.sortingTable ||
item._id === pmcData.Inventory.stash
)
2023-03-03 16:23:46 +01:00
{
continue;
}
}
// replace id
const oldId = item._id;
const newId = this.hashUtil.generate();
serialisedInventory = serialisedInventory.replace(new RegExp(oldId, "g"), newId);
// Also replace in quick slot if the old ID exists.
if (fastPanel !== null)
{
for (const itemSlot in fastPanel)
{
if (fastPanel[itemSlot] === oldId)
{
fastPanel[itemSlot] = fastPanel[itemSlot].replace(new RegExp(oldId, "g"), newId);
}
}
}
}
items = this.jsonUtil.deserialize(serialisedInventory);
// fix duplicate id's
const dupes: Record<string, number> = {};
const newParents: Record<string, Item[]> = {};
const childrenMapping = {};
const oldToNewIds: Record<string, string[]> = {};
// Finding duplicate IDs involves scanning the item three times.
// First scan - Check which ids are duplicated.
// Second scan - Map parents to items.
// Third scan - Resolve IDs.
for (const item of items)
{
dupes[item._id] = (dupes[item._id] || 0) + 1;
}
for (const item of items)
{
// register the parents
if (dupes[item._id] > 1)
{
const newId = this.hashUtil.generate();
newParents[item.parentId] = newParents[item.parentId] || [];
newParents[item.parentId].push(item);
oldToNewIds[item._id] = oldToNewIds[item._id] || [];
oldToNewIds[item._id].push(newId);
}
}
for (const item of items)
{
if (dupes[item._id] > 1)
{
const oldId = item._id;
const newId = oldToNewIds[oldId].splice(0, 1)[0];
item._id = newId;
// Extract one of the children that's also duplicated.
if (oldId in newParents && newParents[oldId].length > 0)
{
childrenMapping[newId] = {};
for (const childIndex in newParents[oldId])
{
// Make sure we haven't already assigned another duplicate child of
// same slot and location to this parent.
const childId = this.getChildId(newParents[oldId][childIndex]);
if (!(childId in childrenMapping[newId]))
{
childrenMapping[newId][childId] = 1;
newParents[oldId][childIndex].parentId = newId;
// Some very fucking sketchy stuff on this childIndex
// No clue wth was that childIndex supposed to be, but its not
newParents[oldId].splice(Number.parseInt(childIndex), 1);
}
}
}
}
}
return items;
}
/**
* WARNING, SLOW. Recursively loop down through an items hierarchy to see if any of the ids match the supplied list, return true if any do
* @param {string} tpl Items tpl to check parents of
* @param {Array} tplsToCheck Tpl values to check if parents of item match
* @returns boolean Match found
2023-03-03 16:23:46 +01:00
*/
public doesItemOrParentsIdMatch(tpl: string, tplsToCheck: string[]): boolean
{
const itemDetails = this.getItem(tpl);
const itemExists = itemDetails[0];
const item = itemDetails[1];
// not an item, drop out
if (!itemExists)
{
return false;
}
// no parent to check
if (!item._parent)
{
return false;
}
// Does templateId match any values in tplsToCheck array
if (tplsToCheck.includes(item._id))
{
return true;
}
// Does the items parent type exist in tplsToCheck array
if (tplsToCheck.includes(item._parent))
{
return true;
}
// check items parent with same method
return this.doesItemOrParentsIdMatch(item._parent, tplsToCheck);
}
/**
* Check if item is quest item
* @param tpl Items tpl to check quest status of
* @returns true if item is flagged as quest item
2023-03-03 16:23:46 +01:00
*/
public isQuestItem(tpl: string): boolean
{
const itemDetails = this.getItem(tpl);
if (itemDetails[0] && itemDetails[1]._props.QuestItem)
{
return true;
}
return false;
}
Refactor of InsuranceController & New ItemHelper Methods (!151) This commit is my second go-around at refactoring the `InsuranceController`, attempting to improving the code's modularity, maintainability, and efficiency while squashing a few bugs along the way. 1. **InsuranceController.ts** - Removed `ITemplateItem` import, as it is no longer used. - Introduced the `adoptOrphanedItems` method to manage orphaned items in the insurance list. - Since "normal" items are individually rolled for deletion, and can be nested within one another, there are situations where a parent item is deleted, leaving its children orphaned. This method moves those orphaned children from their missing parent into the root of the insurance container. - Overhauled `findItemsToDelete` method to improve its efficiency and readability: - Divided the original monolithic method into smaller, specialized methods like `populateItemsMap`, `populateParentAttachmentsMap`, `processRegularItems`, and `processAttachments`. - Changed the return type to `Set<string>` for better performance. - Introduced `EnrichedItem` interface (a simple extension of the `Item` interface) to add additional item data, like `name` and `maxPrice` to `Item` objects as they're processed throughout the class. This is done in place of repeatedly querying for this data, or complicating return types. - Enhanced logging capabilities to debug the item deletion process. Due to the *current* lack of testing available I've stepped up the amount of debug logging that is done. This will hopefully help us find issues in the future. - Modified the `rollForItemDelete` method, now renamed to `rollForDelete`, to include more detailed logging, return a boolean directly, and changed the `insuredItem` parameter to be optional. - Added new methods for dealing with some of the particulars that arise from item adoption and creating item maps. - Improved inline comments and documentation for better code maintainability. 2. **ItemHelper.ts** - Added the `isRaidModdable` method to check if an item is *actually* modifiable in-raid, which takes into account not just the item, but the item that it's attached to. - Added the `getAttachmentMainParent` method to fetch the main parent item of a given attachment, useful for item hierarchy traversal. For example, if you pass it an item ID of a suppressor, it will traverse up the muzzle brake, barrel, upper receiver, and return the gun that the suppressor is ultimately attached to, even if that gun is located within other multiple containers. - Added the `isAttachmentAttached` method to check if an item is an attachment that is currently attached to its parent. **Fixes:** - Resolved an issue that caused item attachments from being property grouped together for deletion rolls. This issue prevented valuable attachments from being taken first. - Resolved an issue that caused child items being orphaned when their parent was removed due to an insurance roll. Probable cause of the bug that made the client spaz out and send repeated insurance packages to the profile---Though I'm still unable to reproduce. - Probably more... Co-authored-by: Refringe <brownelltyler@gmail.com> Reviewed-on: https://dev.sp-tarkov.com/SPT-AKI/Server/pulls/151 Co-authored-by: Refringe <refringe@noreply.dev.sp-tarkov.com> Co-committed-by: Refringe <refringe@noreply.dev.sp-tarkov.com>
2023-10-14 11:05:49 +02:00
/**
* Checks to see if the item is *actually* moddable in-raid. Checks include the items existence in the database, the
* parent items existence in the database, the existence (and value) of the items RaidModdable property, and that
* the parents slot-required property exists, matches that of the item, and it's value.
*
* Note: this function does not preform any checks to see if the item and parent are *actually* related.
*
* @param item The item to be checked
* @param parent The parent of the item to be checked
* @returns True if the item is actually moddable, false if it is not, and null if the check cannot be performed.
*/
public isRaidModdable(item: Item, parent: Item): boolean | null
{
// This check requires the item to have the slotId property populated.
if (!item.slotId)
{
return null;
}
const itemTemplate = this.getItem(item._tpl);
const parentTemplate = this.getItem(parent._tpl);
// Check for RaidModdable property on the item template.
let isNotRaidModdable = false;
if (itemTemplate[0])
{
isNotRaidModdable = itemTemplate[1]?._props?.RaidModdable === false;
}
// Check to see if the slot that the item is attached to is marked as required in the parent item's template.
let isRequiredSlot = false;
if (parentTemplate[0] && parentTemplate[1]?._props?.Slots)
{
2023-11-13 17:07:59 +01:00
isRequiredSlot = parentTemplate[1]._props.Slots.some((slot) =>
slot._name === item.slotId && slot._required
);
Refactor of InsuranceController & New ItemHelper Methods (!151) This commit is my second go-around at refactoring the `InsuranceController`, attempting to improving the code's modularity, maintainability, and efficiency while squashing a few bugs along the way. 1. **InsuranceController.ts** - Removed `ITemplateItem` import, as it is no longer used. - Introduced the `adoptOrphanedItems` method to manage orphaned items in the insurance list. - Since "normal" items are individually rolled for deletion, and can be nested within one another, there are situations where a parent item is deleted, leaving its children orphaned. This method moves those orphaned children from their missing parent into the root of the insurance container. - Overhauled `findItemsToDelete` method to improve its efficiency and readability: - Divided the original monolithic method into smaller, specialized methods like `populateItemsMap`, `populateParentAttachmentsMap`, `processRegularItems`, and `processAttachments`. - Changed the return type to `Set<string>` for better performance. - Introduced `EnrichedItem` interface (a simple extension of the `Item` interface) to add additional item data, like `name` and `maxPrice` to `Item` objects as they're processed throughout the class. This is done in place of repeatedly querying for this data, or complicating return types. - Enhanced logging capabilities to debug the item deletion process. Due to the *current* lack of testing available I've stepped up the amount of debug logging that is done. This will hopefully help us find issues in the future. - Modified the `rollForItemDelete` method, now renamed to `rollForDelete`, to include more detailed logging, return a boolean directly, and changed the `insuredItem` parameter to be optional. - Added new methods for dealing with some of the particulars that arise from item adoption and creating item maps. - Improved inline comments and documentation for better code maintainability. 2. **ItemHelper.ts** - Added the `isRaidModdable` method to check if an item is *actually* modifiable in-raid, which takes into account not just the item, but the item that it's attached to. - Added the `getAttachmentMainParent` method to fetch the main parent item of a given attachment, useful for item hierarchy traversal. For example, if you pass it an item ID of a suppressor, it will traverse up the muzzle brake, barrel, upper receiver, and return the gun that the suppressor is ultimately attached to, even if that gun is located within other multiple containers. - Added the `isAttachmentAttached` method to check if an item is an attachment that is currently attached to its parent. **Fixes:** - Resolved an issue that caused item attachments from being property grouped together for deletion rolls. This issue prevented valuable attachments from being taken first. - Resolved an issue that caused child items being orphaned when their parent was removed due to an insurance roll. Probable cause of the bug that made the client spaz out and send repeated insurance packages to the profile---Though I'm still unable to reproduce. - Probably more... Co-authored-by: Refringe <brownelltyler@gmail.com> Reviewed-on: https://dev.sp-tarkov.com/SPT-AKI/Server/pulls/151 Co-authored-by: Refringe <refringe@noreply.dev.sp-tarkov.com> Co-committed-by: Refringe <refringe@noreply.dev.sp-tarkov.com>
2023-10-14 11:05:49 +02:00
}
return itemTemplate[0] && parentTemplate[0] && !(isNotRaidModdable || isRequiredSlot);
}
/**
* Retrieves the main parent item for a given attachment item.
*
* This method traverses up the hierarchy of items starting from a given `itemId`, until it finds the main parent
* item that is not an attached attachment itself. In other words, if you pass it an item id of a suppressor, it
* will traverse up the muzzle brake, barrel, upper receiver, and return the gun that the suppressor is ultimately
* attached to, even if that gun is located within multiple containers.
*
* It's important to note that traversal is expensive, so this method requires that you pass it a Map of the items
* to traverse, where the keys are the item IDs and the values are the corresponding Item objects. This alleviates
* some of the performance concerns, as it allows for quick lookups of items by ID.
*
* To generate the map:
* ```
* const itemsMap = new Map<string, Item>();
* items.forEach(item => itemsMap.set(item._id, item));
* ```
*
* @param itemId - The unique identifier of the item for which to find the main parent.
* @param itemsMap - A Map containing item IDs mapped to their corresponding Item objects for quick lookup.
* @returns The Item object representing the top-most parent of the given item, or `null` if no such parent exists.
*/
public getAttachmentMainParent(itemId: string, itemsMap: Map<string, Item>): Item | null
{
let currentItem = itemsMap.get(itemId);
while (currentItem && this.isAttachmentAttached(currentItem))
{
currentItem = itemsMap.get(currentItem.parentId);
if (!currentItem)
{
return null;
}
Refactor of InsuranceController & New ItemHelper Methods (!151) This commit is my second go-around at refactoring the `InsuranceController`, attempting to improving the code's modularity, maintainability, and efficiency while squashing a few bugs along the way. 1. **InsuranceController.ts** - Removed `ITemplateItem` import, as it is no longer used. - Introduced the `adoptOrphanedItems` method to manage orphaned items in the insurance list. - Since "normal" items are individually rolled for deletion, and can be nested within one another, there are situations where a parent item is deleted, leaving its children orphaned. This method moves those orphaned children from their missing parent into the root of the insurance container. - Overhauled `findItemsToDelete` method to improve its efficiency and readability: - Divided the original monolithic method into smaller, specialized methods like `populateItemsMap`, `populateParentAttachmentsMap`, `processRegularItems`, and `processAttachments`. - Changed the return type to `Set<string>` for better performance. - Introduced `EnrichedItem` interface (a simple extension of the `Item` interface) to add additional item data, like `name` and `maxPrice` to `Item` objects as they're processed throughout the class. This is done in place of repeatedly querying for this data, or complicating return types. - Enhanced logging capabilities to debug the item deletion process. Due to the *current* lack of testing available I've stepped up the amount of debug logging that is done. This will hopefully help us find issues in the future. - Modified the `rollForItemDelete` method, now renamed to `rollForDelete`, to include more detailed logging, return a boolean directly, and changed the `insuredItem` parameter to be optional. - Added new methods for dealing with some of the particulars that arise from item adoption and creating item maps. - Improved inline comments and documentation for better code maintainability. 2. **ItemHelper.ts** - Added the `isRaidModdable` method to check if an item is *actually* modifiable in-raid, which takes into account not just the item, but the item that it's attached to. - Added the `getAttachmentMainParent` method to fetch the main parent item of a given attachment, useful for item hierarchy traversal. For example, if you pass it an item ID of a suppressor, it will traverse up the muzzle brake, barrel, upper receiver, and return the gun that the suppressor is ultimately attached to, even if that gun is located within other multiple containers. - Added the `isAttachmentAttached` method to check if an item is an attachment that is currently attached to its parent. **Fixes:** - Resolved an issue that caused item attachments from being property grouped together for deletion rolls. This issue prevented valuable attachments from being taken first. - Resolved an issue that caused child items being orphaned when their parent was removed due to an insurance roll. Probable cause of the bug that made the client spaz out and send repeated insurance packages to the profile---Though I'm still unable to reproduce. - Probably more... Co-authored-by: Refringe <brownelltyler@gmail.com> Reviewed-on: https://dev.sp-tarkov.com/SPT-AKI/Server/pulls/151 Co-authored-by: Refringe <refringe@noreply.dev.sp-tarkov.com> Co-committed-by: Refringe <refringe@noreply.dev.sp-tarkov.com>
2023-10-14 11:05:49 +02:00
}
return currentItem;
}
/**
* Determines if an item is an attachment that is currently attached to it's parent item.
*
* @param item The item to check.
* @returns true if the item is attached attachment, otherwise false.
*/
public isAttachmentAttached(item: Item): boolean
{
2023-10-31 23:54:59 +01:00
return item.slotId !== "hideout" && item.slotId !== "main" && Number.isNaN(Number(item.slotId));
Refactor of InsuranceController & New ItemHelper Methods (!151) This commit is my second go-around at refactoring the `InsuranceController`, attempting to improving the code's modularity, maintainability, and efficiency while squashing a few bugs along the way. 1. **InsuranceController.ts** - Removed `ITemplateItem` import, as it is no longer used. - Introduced the `adoptOrphanedItems` method to manage orphaned items in the insurance list. - Since "normal" items are individually rolled for deletion, and can be nested within one another, there are situations where a parent item is deleted, leaving its children orphaned. This method moves those orphaned children from their missing parent into the root of the insurance container. - Overhauled `findItemsToDelete` method to improve its efficiency and readability: - Divided the original monolithic method into smaller, specialized methods like `populateItemsMap`, `populateParentAttachmentsMap`, `processRegularItems`, and `processAttachments`. - Changed the return type to `Set<string>` for better performance. - Introduced `EnrichedItem` interface (a simple extension of the `Item` interface) to add additional item data, like `name` and `maxPrice` to `Item` objects as they're processed throughout the class. This is done in place of repeatedly querying for this data, or complicating return types. - Enhanced logging capabilities to debug the item deletion process. Due to the *current* lack of testing available I've stepped up the amount of debug logging that is done. This will hopefully help us find issues in the future. - Modified the `rollForItemDelete` method, now renamed to `rollForDelete`, to include more detailed logging, return a boolean directly, and changed the `insuredItem` parameter to be optional. - Added new methods for dealing with some of the particulars that arise from item adoption and creating item maps. - Improved inline comments and documentation for better code maintainability. 2. **ItemHelper.ts** - Added the `isRaidModdable` method to check if an item is *actually* modifiable in-raid, which takes into account not just the item, but the item that it's attached to. - Added the `getAttachmentMainParent` method to fetch the main parent item of a given attachment, useful for item hierarchy traversal. For example, if you pass it an item ID of a suppressor, it will traverse up the muzzle brake, barrel, upper receiver, and return the gun that the suppressor is ultimately attached to, even if that gun is located within other multiple containers. - Added the `isAttachmentAttached` method to check if an item is an attachment that is currently attached to its parent. **Fixes:** - Resolved an issue that caused item attachments from being property grouped together for deletion rolls. This issue prevented valuable attachments from being taken first. - Resolved an issue that caused child items being orphaned when their parent was removed due to an insurance roll. Probable cause of the bug that made the client spaz out and send repeated insurance packages to the profile---Though I'm still unable to reproduce. - Probably more... Co-authored-by: Refringe <brownelltyler@gmail.com> Reviewed-on: https://dev.sp-tarkov.com/SPT-AKI/Server/pulls/151 Co-authored-by: Refringe <refringe@noreply.dev.sp-tarkov.com> Co-committed-by: Refringe <refringe@noreply.dev.sp-tarkov.com>
2023-10-14 11:05:49 +02:00
}
2023-03-03 16:23:46 +01:00
/**
* Get the inventory size of an item
* @param items Item with children
Refactor of InsuranceController & New ItemHelper Methods (!151) This commit is my second go-around at refactoring the `InsuranceController`, attempting to improving the code's modularity, maintainability, and efficiency while squashing a few bugs along the way. 1. **InsuranceController.ts** - Removed `ITemplateItem` import, as it is no longer used. - Introduced the `adoptOrphanedItems` method to manage orphaned items in the insurance list. - Since "normal" items are individually rolled for deletion, and can be nested within one another, there are situations where a parent item is deleted, leaving its children orphaned. This method moves those orphaned children from their missing parent into the root of the insurance container. - Overhauled `findItemsToDelete` method to improve its efficiency and readability: - Divided the original monolithic method into smaller, specialized methods like `populateItemsMap`, `populateParentAttachmentsMap`, `processRegularItems`, and `processAttachments`. - Changed the return type to `Set<string>` for better performance. - Introduced `EnrichedItem` interface (a simple extension of the `Item` interface) to add additional item data, like `name` and `maxPrice` to `Item` objects as they're processed throughout the class. This is done in place of repeatedly querying for this data, or complicating return types. - Enhanced logging capabilities to debug the item deletion process. Due to the *current* lack of testing available I've stepped up the amount of debug logging that is done. This will hopefully help us find issues in the future. - Modified the `rollForItemDelete` method, now renamed to `rollForDelete`, to include more detailed logging, return a boolean directly, and changed the `insuredItem` parameter to be optional. - Added new methods for dealing with some of the particulars that arise from item adoption and creating item maps. - Improved inline comments and documentation for better code maintainability. 2. **ItemHelper.ts** - Added the `isRaidModdable` method to check if an item is *actually* modifiable in-raid, which takes into account not just the item, but the item that it's attached to. - Added the `getAttachmentMainParent` method to fetch the main parent item of a given attachment, useful for item hierarchy traversal. For example, if you pass it an item ID of a suppressor, it will traverse up the muzzle brake, barrel, upper receiver, and return the gun that the suppressor is ultimately attached to, even if that gun is located within other multiple containers. - Added the `isAttachmentAttached` method to check if an item is an attachment that is currently attached to its parent. **Fixes:** - Resolved an issue that caused item attachments from being property grouped together for deletion rolls. This issue prevented valuable attachments from being taken first. - Resolved an issue that caused child items being orphaned when their parent was removed due to an insurance roll. Probable cause of the bug that made the client spaz out and send repeated insurance packages to the profile---Though I'm still unable to reproduce. - Probably more... Co-authored-by: Refringe <brownelltyler@gmail.com> Reviewed-on: https://dev.sp-tarkov.com/SPT-AKI/Server/pulls/151 Co-authored-by: Refringe <refringe@noreply.dev.sp-tarkov.com> Co-committed-by: Refringe <refringe@noreply.dev.sp-tarkov.com>
2023-10-14 11:05:49 +02:00
* @param rootItemId
2023-03-03 16:23:46 +01:00
* @returns ItemSize object (width and height)
*/
public getItemSize(items: Item[], rootItemId: string): ItemHelper.ItemSize
{
2023-11-13 17:07:59 +01:00
const rootTemplate = this.getItem(items.filter((x) => x._id === rootItemId)[0]._tpl)[1];
2023-03-03 16:23:46 +01:00
const width = rootTemplate._props.Width;
const height = rootTemplate._props.Height;
let sizeUp = 0;
let sizeDown = 0;
let sizeLeft = 0;
let sizeRight = 0;
let forcedUp = 0;
let forcedDown = 0;
let forcedLeft = 0;
let forcedRight = 0;
const children = this.findAndReturnChildrenAsItems(items, rootItemId);
for (const ci of children)
{
const itemTemplate = this.getItem(ci._tpl)[1];
// Calculating child ExtraSize
if (itemTemplate._props.ExtraSizeForceAdd === true)
{
forcedUp += itemTemplate._props.ExtraSizeUp;
forcedDown += itemTemplate._props.ExtraSizeDown;
forcedLeft += itemTemplate._props.ExtraSizeLeft;
forcedRight += itemTemplate._props.ExtraSizeRight;
}
else
{
sizeUp = sizeUp < itemTemplate._props.ExtraSizeUp ? itemTemplate._props.ExtraSizeUp : sizeUp;
sizeDown = sizeDown < itemTemplate._props.ExtraSizeDown ? itemTemplate._props.ExtraSizeDown : sizeDown;
sizeLeft = sizeLeft < itemTemplate._props.ExtraSizeLeft ? itemTemplate._props.ExtraSizeLeft : sizeLeft;
2023-11-13 17:07:59 +01:00
sizeRight = sizeRight < itemTemplate._props.ExtraSizeRight ?
itemTemplate._props.ExtraSizeRight :
sizeRight;
2023-03-03 16:23:46 +01:00
}
}
return {
width: width + sizeLeft + sizeRight + forcedLeft + forcedRight,
2023-11-13 17:07:59 +01:00
height: height + sizeUp + sizeDown + forcedUp + forcedDown,
2023-03-03 16:23:46 +01:00
};
}
/**
* Get a random cartridge from an items Filter property
* @param item Db item template to look up Cartridge filter values from
* @returns Caliber of cartridge
2023-03-03 16:23:46 +01:00
*/
public getRandomCompatibleCaliberTemplateId(item: ITemplateItem): string | null
2023-03-03 16:23:46 +01:00
{
const cartridges = item?._props?.Cartridges[0]?._props?.filters[0]?.Filter;
2023-03-03 16:23:46 +01:00
if (!cartridges)
{
this.logger.warning(`Failed to find cartridge for item: ${item?._id} ${item?._name}`);
2023-03-03 16:23:46 +01:00
return null;
}
return this.randomUtil.getArrayValue(cartridges);
2023-03-03 16:23:46 +01:00
}
/**
* Add cartridges to the ammo box with correct max stack sizes
* @param ammoBox Box to add cartridges to
* @param ammoBoxDetails Item template from items db
*/
public addCartridgesToAmmoBox(ammoBox: Item[], ammoBoxDetails: ITemplateItem): void
{
const ammoBoxMaxCartridgeCount = ammoBoxDetails._props.StackSlots[0]._max_count;
const cartridgeTpl = ammoBoxDetails._props.StackSlots[0]._props.filters[0].Filter[0];
const cartridgeDetails = this.getItem(cartridgeTpl);
const cartridgeMaxStackSize = cartridgeDetails[1]._props.StackMaxSize;
// Add new stack-size-correct items to ammo box
let currentStoredCartridgeCount = 0;
// Location in ammoBox cartridges will be placed
2023-03-03 16:23:46 +01:00
let location = 0;
const maxPerStack = Math.min(ammoBoxMaxCartridgeCount, cartridgeMaxStackSize);
2023-03-03 16:23:46 +01:00
while (currentStoredCartridgeCount < ammoBoxMaxCartridgeCount)
{
const remainingSpace = ammoBoxMaxCartridgeCount - currentStoredCartridgeCount;
2023-11-13 17:07:59 +01:00
const cartridgeCountToAdd = (remainingSpace < maxPerStack) ?
remainingSpace :
maxPerStack;
2023-03-03 16:23:46 +01:00
// Add cartridge item into items array
2023-03-03 16:23:46 +01:00
ammoBox.push(this.createCartridges(ammoBox[0]._id, cartridgeTpl, cartridgeCountToAdd, location));
currentStoredCartridgeCount += cartridgeCountToAdd;
2023-11-13 17:07:59 +01:00
location++;
2023-03-03 16:23:46 +01:00
}
}
/**
* Check if item is stored inside of a container
* @param item Item to check is inside of container
* @param desiredContainerSlotId Name of slot to check item is in e.g. SecuredContainer/Backpack
* @param items Inventory with child parent items to check
* @returns True when item is in container
*/
public itemIsInsideContainer(item: Item, desiredContainerSlotId: string, items: Item[]): boolean
{
// Get items parent
2023-11-13 17:07:59 +01:00
const parent = items.find((x) => x._id === item.parentId);
if (!parent)
{
// No parent, end of line, not inside container
return false;
}
if (parent.slotId === desiredContainerSlotId)
{
return true;
}
else
{
return this.itemIsInsideContainer(parent, desiredContainerSlotId, items);
}
}
/**
* Add child items (cartridges) to a magazine
* @param magazine Magazine to add child items to
* @param magTemplate Db template of magazine
* @param staticAmmoDist Cartridge distribution
* @param caliber Caliber of cartridge to add to magazine
* @param minSizePercent % the magazine must be filled to
*/
public fillMagazineWithRandomCartridge(
magazine: Item[],
2023-03-03 16:23:46 +01:00
magTemplate: ITemplateItem,
staticAmmoDist: Record<string, IStaticAmmoDetails[]>,
caliber: string = undefined,
2023-11-13 17:07:59 +01:00
minSizePercent = 0.25,
): void
2023-03-03 16:23:46 +01:00
{
// no caliber defined, choose one at random
2023-03-03 16:23:46 +01:00
if (!caliber)
{
caliber = this.getRandomValidCaliber(magTemplate);
}
// Edge case for the Klin pp-9, it has a typo in its ammo caliber
if (caliber === "Caliber9x18PMM")
{
caliber = "Caliber9x18PM";
}
// Chose a randomly weighted cartridge that fits
const cartridgeTpl = this.drawAmmoTpl(caliber, staticAmmoDist);
this.fillMagazineWithCartridge(magazine, magTemplate, cartridgeTpl, minSizePercent);
}
/**
* Add child items to a magazine of a specific cartridge
* @param magazine Magazine to add child items to
* @param magTemplate Db template of magazine
* @param cartridgeTpl Cartridge to add to magazine
* @param minSizePercent % the magazine must be filled to
*/
public fillMagazineWithCartridge(
magazine: Item[],
magTemplate: ITemplateItem,
cartridgeTpl: string,
2023-11-13 17:07:59 +01:00
minSizePercent = 0.25,
): void
{
// Get cartrdge properties and max allowed stack size
const cartridgeDetails = this.getItem(cartridgeTpl);
const cartridgeMaxStackSize = cartridgeDetails[1]._props.StackMaxSize;
// Get max number of cartridges in magazine, choose random value between min/max
const magazineCartridgeMaxCount = magTemplate._props.Cartridges[0]._max_count;
2023-11-13 17:07:59 +01:00
const desiredStackCount = this.randomUtil.getInt(
Math.round(minSizePercent * magazineCartridgeMaxCount),
magazineCartridgeMaxCount,
);
// Loop over cartridge count and add stacks to magazine
let currentStoredCartridgeCount = 0;
let location = 0;
while (currentStoredCartridgeCount < desiredStackCount)
{
// Get stack size of cartridges
2023-11-13 17:07:59 +01:00
let cartridgeCountToAdd = (desiredStackCount <= cartridgeMaxStackSize) ?
desiredStackCount :
cartridgeMaxStackSize;
// Ensure we don't go over the max stackcount size
const remainingSpace = desiredStackCount - currentStoredCartridgeCount;
if (cartridgeCountToAdd > remainingSpace)
{
cartridgeCountToAdd = remainingSpace;
}
// Add cartridge item object into items array
magazine.push(this.createCartridges(magazine[0]._id, cartridgeTpl, cartridgeCountToAdd, location));
currentStoredCartridgeCount += cartridgeCountToAdd;
2023-11-13 17:07:59 +01:00
location++;
}
2023-03-03 16:23:46 +01:00
}
/**
* Choose a random bullet type from the list of possible a magazine has
* @param magTemplate Magazine template from Db
* @returns Tpl of cartridge
*/
2023-03-03 16:23:46 +01:00
protected getRandomValidCaliber(magTemplate: ITemplateItem): string
{
const ammoTpls = magTemplate._props.Cartridges[0]._props.filters[0].Filter;
const calibers = [
...new Set(
ammoTpls.filter(
2023-11-13 17:07:59 +01:00
(x: string) => this.getItem(x)[0],
2023-03-03 16:23:46 +01:00
).map(
2023-11-13 17:07:59 +01:00
(x: string) => this.getItem(x)[1]._props.Caliber,
),
),
2023-03-03 16:23:46 +01:00
];
return this.randomUtil.drawRandomFromList(calibers)[0];
}
/**
* Chose a randomly weighted cartridge that fits
* @param caliber Desired caliber
* @param staticAmmoDist Cartridges and thier weights
* @returns Tpl of cartrdige
*/
2023-03-03 16:23:46 +01:00
protected drawAmmoTpl(caliber: string, staticAmmoDist: Record<string, IStaticAmmoDetails[]>): string
{
const ammoArray = new ProbabilityObjectArray<string>(this.mathUtil, this.jsonUtil);
2023-03-03 16:23:46 +01:00
for (const icd of staticAmmoDist[caliber])
{
ammoArray.push(
2023-11-13 17:07:59 +01:00
new ProbabilityObject(icd.tpl, icd.relativeProbability),
2023-03-03 16:23:46 +01:00
);
}
return ammoArray.draw(1)[0];
}
/**
* Create a basic cartrige object
2023-03-03 16:23:46 +01:00
* @param parentId container cartridges will be placed in
* @param ammoTpl Cartridge to insert
Refactor of InsuranceController & New ItemHelper Methods (!151) This commit is my second go-around at refactoring the `InsuranceController`, attempting to improving the code's modularity, maintainability, and efficiency while squashing a few bugs along the way. 1. **InsuranceController.ts** - Removed `ITemplateItem` import, as it is no longer used. - Introduced the `adoptOrphanedItems` method to manage orphaned items in the insurance list. - Since "normal" items are individually rolled for deletion, and can be nested within one another, there are situations where a parent item is deleted, leaving its children orphaned. This method moves those orphaned children from their missing parent into the root of the insurance container. - Overhauled `findItemsToDelete` method to improve its efficiency and readability: - Divided the original monolithic method into smaller, specialized methods like `populateItemsMap`, `populateParentAttachmentsMap`, `processRegularItems`, and `processAttachments`. - Changed the return type to `Set<string>` for better performance. - Introduced `EnrichedItem` interface (a simple extension of the `Item` interface) to add additional item data, like `name` and `maxPrice` to `Item` objects as they're processed throughout the class. This is done in place of repeatedly querying for this data, or complicating return types. - Enhanced logging capabilities to debug the item deletion process. Due to the *current* lack of testing available I've stepped up the amount of debug logging that is done. This will hopefully help us find issues in the future. - Modified the `rollForItemDelete` method, now renamed to `rollForDelete`, to include more detailed logging, return a boolean directly, and changed the `insuredItem` parameter to be optional. - Added new methods for dealing with some of the particulars that arise from item adoption and creating item maps. - Improved inline comments and documentation for better code maintainability. 2. **ItemHelper.ts** - Added the `isRaidModdable` method to check if an item is *actually* modifiable in-raid, which takes into account not just the item, but the item that it's attached to. - Added the `getAttachmentMainParent` method to fetch the main parent item of a given attachment, useful for item hierarchy traversal. For example, if you pass it an item ID of a suppressor, it will traverse up the muzzle brake, barrel, upper receiver, and return the gun that the suppressor is ultimately attached to, even if that gun is located within other multiple containers. - Added the `isAttachmentAttached` method to check if an item is an attachment that is currently attached to its parent. **Fixes:** - Resolved an issue that caused item attachments from being property grouped together for deletion rolls. This issue prevented valuable attachments from being taken first. - Resolved an issue that caused child items being orphaned when their parent was removed due to an insurance roll. Probable cause of the bug that made the client spaz out and send repeated insurance packages to the profile---Though I'm still unable to reproduce. - Probably more... Co-authored-by: Refringe <brownelltyler@gmail.com> Reviewed-on: https://dev.sp-tarkov.com/SPT-AKI/Server/pulls/151 Co-authored-by: Refringe <refringe@noreply.dev.sp-tarkov.com> Co-committed-by: Refringe <refringe@noreply.dev.sp-tarkov.com>
2023-10-14 11:05:49 +02:00
* @param stackCount Count of cartridges inside parent
2023-03-03 16:23:46 +01:00
* @param location Location inside parent (e.g. 0, 1)
* @returns Item
*/
public createCartridges(parentId: string, ammoTpl: string, stackCount: number, location: number): Item
{
return {
_id: this.objectId.generate(),
_tpl: ammoTpl,
parentId: parentId,
slotId: "cartridges",
location: location,
2023-11-13 17:07:59 +01:00
upd: {StackObjectsCount: stackCount},
2023-03-03 16:23:46 +01:00
};
}
/**
* Get the size of a stack, return 1 if no stack object count property found
Refactor of InsuranceController & New ItemHelper Methods (!151) This commit is my second go-around at refactoring the `InsuranceController`, attempting to improving the code's modularity, maintainability, and efficiency while squashing a few bugs along the way. 1. **InsuranceController.ts** - Removed `ITemplateItem` import, as it is no longer used. - Introduced the `adoptOrphanedItems` method to manage orphaned items in the insurance list. - Since "normal" items are individually rolled for deletion, and can be nested within one another, there are situations where a parent item is deleted, leaving its children orphaned. This method moves those orphaned children from their missing parent into the root of the insurance container. - Overhauled `findItemsToDelete` method to improve its efficiency and readability: - Divided the original monolithic method into smaller, specialized methods like `populateItemsMap`, `populateParentAttachmentsMap`, `processRegularItems`, and `processAttachments`. - Changed the return type to `Set<string>` for better performance. - Introduced `EnrichedItem` interface (a simple extension of the `Item` interface) to add additional item data, like `name` and `maxPrice` to `Item` objects as they're processed throughout the class. This is done in place of repeatedly querying for this data, or complicating return types. - Enhanced logging capabilities to debug the item deletion process. Due to the *current* lack of testing available I've stepped up the amount of debug logging that is done. This will hopefully help us find issues in the future. - Modified the `rollForItemDelete` method, now renamed to `rollForDelete`, to include more detailed logging, return a boolean directly, and changed the `insuredItem` parameter to be optional. - Added new methods for dealing with some of the particulars that arise from item adoption and creating item maps. - Improved inline comments and documentation for better code maintainability. 2. **ItemHelper.ts** - Added the `isRaidModdable` method to check if an item is *actually* modifiable in-raid, which takes into account not just the item, but the item that it's attached to. - Added the `getAttachmentMainParent` method to fetch the main parent item of a given attachment, useful for item hierarchy traversal. For example, if you pass it an item ID of a suppressor, it will traverse up the muzzle brake, barrel, upper receiver, and return the gun that the suppressor is ultimately attached to, even if that gun is located within other multiple containers. - Added the `isAttachmentAttached` method to check if an item is an attachment that is currently attached to its parent. **Fixes:** - Resolved an issue that caused item attachments from being property grouped together for deletion rolls. This issue prevented valuable attachments from being taken first. - Resolved an issue that caused child items being orphaned when their parent was removed due to an insurance roll. Probable cause of the bug that made the client spaz out and send repeated insurance packages to the profile---Though I'm still unable to reproduce. - Probably more... Co-authored-by: Refringe <brownelltyler@gmail.com> Reviewed-on: https://dev.sp-tarkov.com/SPT-AKI/Server/pulls/151 Co-authored-by: Refringe <refringe@noreply.dev.sp-tarkov.com> Co-committed-by: Refringe <refringe@noreply.dev.sp-tarkov.com>
2023-10-14 11:05:49 +02:00
* @param item Item to get stack size of
2023-03-03 16:23:46 +01:00
* @returns size of stack
*/
public getItemStackSize(item: Item): number
{
if (item.upd?.StackObjectsCount)
{
return item.upd.StackObjectsCount;
}
return 1;
}
/**
* Get the name of an item from the locale file using the item tpl
* @param itemTpl Tpl of item to get name of
* @returns Name of item
*/
public getItemName(itemTpl: string): string
{
return this.localeService.getLocaleDb()[`${itemTpl} Name`];
}
public getItemTplsOfBaseType(desiredBaseType: string): string[]
{
2023-11-13 17:07:59 +01:00
return Object.values(this.databaseServer.getTables().templates.items).filter((x) =>
x._parent === desiredBaseType
).map((x) => x._id);
}
2023-03-03 16:23:46 +01:00
}
namespace ItemHelper
{
export interface ItemSize
{
2023-11-13 17:07:59 +01:00
width: number;
height: number;
2023-03-03 16:23:46 +01:00
}
}
2023-11-13 17:07:59 +01:00
export {ItemHelper};