2023-03-03 16:23:46 +01:00
|
|
|
import { inject, injectable } from "tsyringe";
|
|
|
|
|
2023-10-19 19:21:17 +02:00
|
|
|
import { HandbookHelper } from "@spt-aki/helpers/HandbookHelper";
|
|
|
|
import { ItemHelper } from "@spt-aki/helpers/ItemHelper";
|
2024-01-20 17:20:39 +01:00
|
|
|
import { PresetHelper } from "@spt-aki/helpers/PresetHelper";
|
2023-10-19 19:21:17 +02:00
|
|
|
import { Item } from "@spt-aki/models/eft/common/tables/IItem";
|
|
|
|
import { ITemplateItem } from "@spt-aki/models/eft/common/tables/ITemplateItem";
|
|
|
|
import { IBarterScheme } from "@spt-aki/models/eft/common/tables/ITrader";
|
|
|
|
import { ConfigTypes } from "@spt-aki/models/enums/ConfigTypes";
|
|
|
|
import { Money } from "@spt-aki/models/enums/Money";
|
|
|
|
import { Traders } from "@spt-aki/models/enums/Traders";
|
|
|
|
import { ITraderConfig } from "@spt-aki/models/spt/config/ITraderConfig";
|
|
|
|
import { ILogger } from "@spt-aki/models/spt/utils/ILogger";
|
|
|
|
import { ConfigServer } from "@spt-aki/servers/ConfigServer";
|
|
|
|
import { DatabaseServer } from "@spt-aki/servers/DatabaseServer";
|
|
|
|
import { ItemFilterService } from "@spt-aki/services/ItemFilterService";
|
|
|
|
import { SeasonalEventService } from "@spt-aki/services/SeasonalEventService";
|
2024-01-20 17:20:39 +01:00
|
|
|
import { HashUtil } from "@spt-aki/utils/HashUtil";
|
|
|
|
import { JsonUtil } from "@spt-aki/utils/JsonUtil";
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
@injectable()
|
|
|
|
export class FenceBaseAssortGenerator
|
|
|
|
{
|
|
|
|
protected traderConfig: ITraderConfig;
|
|
|
|
|
|
|
|
constructor(
|
|
|
|
@inject("WinstonLogger") protected logger: ILogger,
|
2024-01-20 17:20:39 +01:00
|
|
|
@inject("HashUtil") protected hashUtil: HashUtil,
|
|
|
|
@inject("JsonUtil") protected jsonUtil: JsonUtil,
|
2023-03-03 16:23:46 +01:00
|
|
|
@inject("DatabaseServer") protected databaseServer: DatabaseServer,
|
|
|
|
@inject("HandbookHelper") protected handbookHelper: HandbookHelper,
|
|
|
|
@inject("ItemHelper") protected itemHelper: ItemHelper,
|
2024-01-20 17:20:39 +01:00
|
|
|
@inject("PresetHelper") protected presetHelper: PresetHelper,
|
2023-03-03 16:23:46 +01:00
|
|
|
@inject("ItemFilterService") protected itemFilterService: ItemFilterService,
|
2023-07-22 15:11:47 +02:00
|
|
|
@inject("SeasonalEventService") protected seasonalEventService: SeasonalEventService,
|
2023-11-16 02:35:05 +01:00
|
|
|
@inject("ConfigServer") protected configServer: ConfigServer,
|
2023-03-03 16:23:46 +01:00
|
|
|
)
|
|
|
|
{
|
|
|
|
this.traderConfig = this.configServer.getConfig(ConfigTypes.TRADER);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create base fence assorts dynamically and store in db
|
|
|
|
*/
|
|
|
|
public generateFenceBaseAssorts(): void
|
|
|
|
{
|
2023-11-01 12:35:05 +01:00
|
|
|
const blockedSeasonalItems = this.seasonalEventService.getInactiveSeasonalEventItems();
|
2023-03-03 16:23:46 +01:00
|
|
|
const baseFenceAssort = this.databaseServer.getTables().traders[Traders.FENCE].assort;
|
|
|
|
|
2024-01-20 17:20:39 +01:00
|
|
|
for (const rootItemDb of this.itemHelper.getItems().filter((item) => this.isValidFenceItem(item)))
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
// Skip blacklisted items
|
2024-01-20 17:20:39 +01:00
|
|
|
if (this.itemFilterService.isItemBlacklisted(rootItemDb._id))
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2024-01-20 17:20:39 +01:00
|
|
|
// Invalid
|
|
|
|
if (!this.itemHelper.isValidItem(rootItemDb._id))
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2024-01-20 17:20:39 +01:00
|
|
|
// Item base type blacklisted
|
2023-03-03 16:23:46 +01:00
|
|
|
if (this.traderConfig.fence.blacklist.length > 0)
|
|
|
|
{
|
2024-01-20 17:20:39 +01:00
|
|
|
if (this.traderConfig.fence.blacklist.includes(rootItemDb._id)
|
|
|
|
|| this.itemHelper.isOfBaseclasses(rootItemDb._id, this.traderConfig.fence.blacklist)
|
2023-11-16 02:35:05 +01:00
|
|
|
)
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-22 15:11:47 +02:00
|
|
|
// Skip seasonal event items when not in seasonal event
|
2024-01-20 17:20:39 +01:00
|
|
|
if (this.traderConfig.fence.blacklistSeasonalItems && blockedSeasonalItems.includes(rootItemDb._id))
|
2023-07-22 15:11:47 +02:00
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2024-01-20 17:20:39 +01:00
|
|
|
// Create item object in array
|
|
|
|
const itemWithChildrenToAdd: Item[] = [{
|
|
|
|
_id: this.hashUtil.generate(),
|
|
|
|
_tpl: rootItemDb._id,
|
|
|
|
parentId: "hideout",
|
|
|
|
slotId: "hideout",
|
|
|
|
upd: { StackObjectsCount: 9999999, UnlimitedCount: true },
|
|
|
|
}];
|
|
|
|
|
|
|
|
// Need to add mods to armors so they dont show as red in the trade screen
|
|
|
|
let price = this.handbookHelper.getTemplatePrice(rootItemDb._id);
|
|
|
|
if (this.itemHelper.itemRequiresSoftInserts(rootItemDb._id))
|
|
|
|
{
|
|
|
|
this.addChildrenToArmorModSlots(itemWithChildrenToAdd, rootItemDb);
|
|
|
|
price = this.getHandbookItemPriceWithChildren(itemWithChildrenToAdd);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure IDs are unique
|
|
|
|
this.itemHelper.remapRootItemId(itemWithChildrenToAdd);
|
|
|
|
if (itemWithChildrenToAdd.length > 1)
|
|
|
|
{
|
|
|
|
this.itemHelper.reparentItemAndChildren(itemWithChildrenToAdd[0], itemWithChildrenToAdd);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create barter scheme (price)
|
2023-03-03 16:23:46 +01:00
|
|
|
const barterSchemeToAdd: IBarterScheme = {
|
2023-11-16 02:35:05 +01:00
|
|
|
count: Math.round(
|
2024-01-20 17:20:39 +01:00
|
|
|
this.handbookHelper.getTemplatePrice(rootItemDb._id) * this.traderConfig.fence.itemPriceMult,
|
2023-11-16 02:35:05 +01:00
|
|
|
),
|
|
|
|
_tpl: Money.ROUBLES,
|
2023-03-03 16:23:46 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
// Add barter data to base
|
2024-01-20 17:20:39 +01:00
|
|
|
baseFenceAssort.barter_scheme[itemWithChildrenToAdd[0]._id] = [[barterSchemeToAdd]];
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
// Add item to base
|
2024-01-20 17:20:39 +01:00
|
|
|
baseFenceAssort.items.push(...itemWithChildrenToAdd);
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
// Add loyalty data to base
|
2024-01-20 17:20:39 +01:00
|
|
|
baseFenceAssort.loyal_level_items[itemWithChildrenToAdd[0]._id] = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add all default presets to base fence assort
|
|
|
|
const defaultPresets = Object.values(this.presetHelper.getDefaultPresets());
|
|
|
|
for (const defaultPreset of defaultPresets)
|
|
|
|
{
|
|
|
|
// Skip presets we've already added
|
|
|
|
if (baseFenceAssort.items.some((item) => item.upd && item.upd.sptPresetId === defaultPreset._id))
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Construct preset + mods
|
|
|
|
const presetAndMods: Item[] = this.itemHelper.replaceIDs(
|
|
|
|
null,
|
|
|
|
this.jsonUtil.clone(defaultPreset._items),
|
|
|
|
);
|
|
|
|
|
|
|
|
for (let i = 0; i < presetAndMods.length; i++)
|
|
|
|
{
|
|
|
|
const mod = presetAndMods[i];
|
|
|
|
|
|
|
|
// Build root Item info
|
|
|
|
if (!("parentId" in mod))
|
|
|
|
{
|
|
|
|
mod._id = presetAndMods[0]._id;
|
|
|
|
mod.parentId = "hideout";
|
|
|
|
mod.slotId = "hideout";
|
|
|
|
mod.upd = {
|
|
|
|
UnlimitedCount: false,
|
|
|
|
StackObjectsCount: 1,
|
|
|
|
BuyRestrictionCurrent: 0,
|
|
|
|
sptPresetId: defaultPreset._id, // Store preset id here so we can check it later to prevent preset dupes
|
|
|
|
};
|
|
|
|
|
|
|
|
// Updated root item, exit loop
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const presetDbItem = this.itemHelper.getItem(presetAndMods[0]._tpl)[1];
|
|
|
|
|
|
|
|
// Add constructed preset to assorts
|
|
|
|
baseFenceAssort.items.push(...presetAndMods);
|
|
|
|
|
|
|
|
// Calculate preset price
|
|
|
|
let rub = 0;
|
|
|
|
for (const it of presetAndMods)
|
|
|
|
{
|
|
|
|
rub += this.handbookHelper.getTemplatePrice(it._tpl);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Multiply weapon+mods rouble price by multipler in config
|
|
|
|
baseFenceAssort.barter_scheme[presetAndMods[0]._id] = [[]];
|
|
|
|
baseFenceAssort.barter_scheme[presetAndMods[0]._id][0][0] = { _tpl: Money.ROUBLES, count: Math.round(rub) };
|
|
|
|
|
|
|
|
baseFenceAssort.loyal_level_items[presetAndMods[0]._id] = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add soft inserts + armor plates to an armor
|
|
|
|
* @param armor Armor item array to add mods into
|
|
|
|
* @param itemDbDetails Armor items db template
|
|
|
|
*/
|
|
|
|
protected addChildrenToArmorModSlots(armor: Item[], itemDbDetails: ITemplateItem): void
|
|
|
|
{
|
|
|
|
// Armor has no mods, make no additions
|
|
|
|
const hasMods = itemDbDetails._props.Slots.length > 0;
|
|
|
|
if (!hasMods)
|
|
|
|
{
|
|
|
|
return;
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
2024-01-20 17:20:39 +01:00
|
|
|
|
|
|
|
// Check for and add required soft inserts to armors
|
|
|
|
const requiredSlots = itemDbDetails._props.Slots.filter(slot => slot._required);
|
|
|
|
const hasRequiredSlots = requiredSlots.length > 0;
|
|
|
|
if (hasRequiredSlots)
|
|
|
|
{
|
|
|
|
for (const requiredSlot of requiredSlots)
|
|
|
|
{
|
|
|
|
const modItemDbDetails = this.itemHelper.getItem(requiredSlot._props.filters[0].Plate)[1];
|
|
|
|
const plateTpl = requiredSlot._props.filters[0].Plate; // `Plate` property appears to be the 'default' item for slot
|
|
|
|
if (plateTpl === "")
|
|
|
|
{
|
|
|
|
// Some bsg plate properties are empty, skip mod
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
const mod: Item = {
|
|
|
|
_id: this.hashUtil.generate(),
|
|
|
|
_tpl: plateTpl,
|
|
|
|
parentId: armor[0]._id,
|
|
|
|
slotId: requiredSlot._name,
|
|
|
|
upd: {
|
|
|
|
Repairable: {
|
|
|
|
Durability: modItemDbDetails._props.MaxDurability,
|
|
|
|
MaxDurability: modItemDbDetails._props.MaxDurability
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
armor.push(mod);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check for and add plate items
|
|
|
|
const plateSlots = itemDbDetails._props.Slots.filter(slot => this.itemHelper.isRemovablePlateSlot(slot._name));
|
|
|
|
if (plateSlots.length > 0)
|
|
|
|
{
|
|
|
|
for (const plateSlot of plateSlots)
|
|
|
|
{
|
|
|
|
const plateTpl = plateSlot._props.filters[0].Plate
|
|
|
|
if (!plateTpl)
|
|
|
|
{
|
|
|
|
// Bsg data lacks a default plate, skip adding mod
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
const modItemDbDetails = this.itemHelper.getItem(plateTpl)[1];
|
|
|
|
armor.push({
|
|
|
|
_id: this.hashUtil.generate(),
|
|
|
|
_tpl: plateSlot._props.filters[0].Plate, // `Plate` property appears to be the 'default' item for slot
|
|
|
|
parentId: armor[0]._id,
|
|
|
|
slotId: plateSlot._name,
|
|
|
|
upd: {
|
|
|
|
Repairable: {
|
|
|
|
Durability: modItemDbDetails._props.MaxDurability,
|
|
|
|
MaxDurability: modItemDbDetails._props.MaxDurability
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Calculate and return the price of an item and its child mods
|
|
|
|
* @param itemWithChildren Item + mods to calcualte price of
|
|
|
|
* @returns price
|
|
|
|
*/
|
|
|
|
protected getHandbookItemPriceWithChildren(itemWithChildren: Item[]): number
|
|
|
|
{
|
|
|
|
let price = 0;
|
|
|
|
for (const item of itemWithChildren)
|
|
|
|
{
|
|
|
|
price += this.handbookHelper.getTemplatePrice(item._tpl);
|
|
|
|
}
|
|
|
|
|
|
|
|
return price;
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check if item is valid for being added to fence assorts
|
|
|
|
* @param item Item to check
|
|
|
|
* @returns true if valid fence item
|
|
|
|
*/
|
|
|
|
protected isValidFenceItem(item: ITemplateItem): boolean
|
|
|
|
{
|
|
|
|
if (item._type === "Item")
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
2023-11-16 02:35:05 +01:00
|
|
|
}
|