2023-03-03 16:23:46 +01:00
|
|
|
import { inject, injectable } from "tsyringe";
|
|
|
|
|
2023-10-19 19:21:17 +02:00
|
|
|
import { InventoryHelper } from "@spt-aki/helpers/InventoryHelper";
|
|
|
|
import { ItemHelper } from "@spt-aki/helpers/ItemHelper";
|
|
|
|
import { PresetHelper } from "@spt-aki/helpers/PresetHelper";
|
|
|
|
import { WeightedRandomHelper } from "@spt-aki/helpers/WeightedRandomHelper";
|
|
|
|
import { IPreset } from "@spt-aki/models/eft/common/IGlobals";
|
|
|
|
import { ITemplateItem } from "@spt-aki/models/eft/common/tables/ITemplateItem";
|
|
|
|
import { AddItem } from "@spt-aki/models/eft/inventory/IAddItemRequestData";
|
|
|
|
import { BaseClasses } from "@spt-aki/models/enums/BaseClasses";
|
|
|
|
import { ISealedAirdropContainerSettings, RewardDetails } from "@spt-aki/models/spt/config/IInventoryConfig";
|
|
|
|
import { LootItem } from "@spt-aki/models/spt/services/LootItem";
|
|
|
|
import { LootRequest } from "@spt-aki/models/spt/services/LootRequest";
|
|
|
|
import { ILogger } from "@spt-aki/models/spt/utils/ILogger";
|
|
|
|
import { DatabaseServer } from "@spt-aki/servers/DatabaseServer";
|
|
|
|
import { ItemFilterService } from "@spt-aki/services/ItemFilterService";
|
|
|
|
import { LocalisationService } from "@spt-aki/services/LocalisationService";
|
|
|
|
import { RagfairLinkedItemService } from "@spt-aki/services/RagfairLinkedItemService";
|
|
|
|
import { HashUtil } from "@spt-aki/utils/HashUtil";
|
|
|
|
import { RandomUtil } from "@spt-aki/utils/RandomUtil";
|
2023-03-03 16:23:46 +01:00
|
|
|
|
2023-04-23 00:41:04 +02:00
|
|
|
type ItemLimit = {
|
2023-11-13 17:05:05 +01:00
|
|
|
current: number;
|
|
|
|
max: number;
|
2023-04-23 00:41:04 +02:00
|
|
|
};
|
|
|
|
|
2023-03-03 16:23:46 +01:00
|
|
|
@injectable()
|
|
|
|
export class LootGenerator
|
|
|
|
{
|
|
|
|
constructor(
|
|
|
|
@inject("WinstonLogger") protected logger: ILogger,
|
|
|
|
@inject("HashUtil") protected hashUtil: HashUtil,
|
|
|
|
@inject("DatabaseServer") protected databaseServer: DatabaseServer,
|
|
|
|
@inject("RandomUtil") protected randomUtil: RandomUtil,
|
|
|
|
@inject("ItemHelper") protected itemHelper: ItemHelper,
|
2023-06-20 17:07:05 +02:00
|
|
|
@inject("PresetHelper") protected presetHelper: PresetHelper,
|
|
|
|
@inject("InventoryHelper") protected inventoryHelper: InventoryHelper,
|
|
|
|
@inject("WeightedRandomHelper") protected weightedRandomHelper: WeightedRandomHelper,
|
2023-03-03 16:23:46 +01:00
|
|
|
@inject("LocalisationService") protected localisationService: LocalisationService,
|
2023-06-20 17:07:05 +02:00
|
|
|
@inject("RagfairLinkedItemService") protected ragfairLinkedItemService: RagfairLinkedItemService,
|
2023-11-13 17:05:05 +01:00
|
|
|
@inject("ItemFilterService") protected itemFilterService: ItemFilterService,
|
2023-03-03 16:23:46 +01:00
|
|
|
)
|
|
|
|
{}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Generate a list of items based on configuration options parameter
|
|
|
|
* @param options parameters to adjust how loot is generated
|
|
|
|
* @returns An array of loot items
|
|
|
|
*/
|
2023-03-18 18:29:26 +01:00
|
|
|
public createRandomLoot(options: LootRequest): LootItem[]
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
const result: LootItem[] = [];
|
|
|
|
|
|
|
|
const itemTypeCounts = this.initItemLimitCounter(options.itemLimits);
|
|
|
|
|
|
|
|
const tables = this.databaseServer.getTables();
|
2023-11-13 17:05:05 +01:00
|
|
|
const itemBlacklist = new Set<string>([
|
|
|
|
...this.itemFilterService.getBlacklistedItems(),
|
|
|
|
...options.itemBlacklist,
|
|
|
|
]);
|
2023-10-12 12:00:04 +02:00
|
|
|
if (!options.allowBossItems)
|
|
|
|
{
|
2023-10-31 18:46:14 +01:00
|
|
|
for (const bossItem of this.itemFilterService.getBossItems())
|
|
|
|
{
|
|
|
|
itemBlacklist.add(bossItem);
|
|
|
|
}
|
2023-10-12 12:00:04 +02:00
|
|
|
}
|
|
|
|
|
2023-06-20 17:59:15 +02:00
|
|
|
// Handle sealed weapon containers
|
2023-11-13 17:05:05 +01:00
|
|
|
const desiredWeaponCrateCount = this.randomUtil.getInt(
|
|
|
|
options.weaponCrateCount.min,
|
|
|
|
options.weaponCrateCount.max,
|
|
|
|
);
|
2023-06-20 17:59:15 +02:00
|
|
|
if (desiredWeaponCrateCount > 0)
|
|
|
|
{
|
|
|
|
// Get list of all sealed containers from db
|
2023-11-13 17:05:05 +01:00
|
|
|
const sealedWeaponContainerPool = Object.values(tables.templates.items).filter((x) =>
|
|
|
|
x._name.includes("event_container_airdrop")
|
|
|
|
);
|
|
|
|
|
2023-06-20 17:59:15 +02:00
|
|
|
for (let index = 0; index < desiredWeaponCrateCount; index++)
|
|
|
|
{
|
|
|
|
// Choose one at random + add to results array
|
|
|
|
const chosenSealedContainer = this.randomUtil.getArrayValue(sealedWeaponContainerPool);
|
|
|
|
result.push({
|
|
|
|
id: this.hashUtil.generate(),
|
|
|
|
tpl: chosenSealedContainer._id,
|
|
|
|
isPreset: false,
|
2023-11-13 17:05:05 +01:00
|
|
|
stackCount: 1,
|
2023-06-20 17:59:15 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-03 16:23:46 +01:00
|
|
|
// Get items from items.json that have a type of item + not in global blacklist + basetype is in whitelist
|
2023-11-13 17:05:05 +01:00
|
|
|
const items = Object.entries(tables.templates.items).filter((x) =>
|
|
|
|
!itemBlacklist.has(x[1]._id) &&
|
|
|
|
x[1]._type.toLowerCase() === "item" &&
|
|
|
|
!x[1]._props.QuestItem &&
|
|
|
|
options.itemTypeWhitelist.includes(x[1]._parent)
|
|
|
|
);
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
const randomisedItemCount = this.randomUtil.getInt(options.itemCount.min, options.itemCount.max);
|
|
|
|
for (let index = 0; index < randomisedItemCount; index++)
|
|
|
|
{
|
|
|
|
if (!this.findAndAddRandomItemToLoot(items, itemTypeCounts, options, result))
|
|
|
|
{
|
|
|
|
index--;
|
2023-11-13 17:05:05 +01:00
|
|
|
}
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
2023-11-13 17:05:05 +01:00
|
|
|
const globalDefaultPresets = Object.entries(tables.globals.ItemPresets).filter((x) =>
|
|
|
|
x[1]._encyclopedia !== undefined
|
|
|
|
);
|
2023-03-03 16:23:46 +01:00
|
|
|
const randomisedPresetCount = this.randomUtil.getInt(options.presetCount.min, options.presetCount.max);
|
2023-10-13 21:18:39 +02:00
|
|
|
const itemBlacklistArray = Array.from(itemBlacklist);
|
2023-03-03 16:23:46 +01:00
|
|
|
for (let index = 0; index < randomisedPresetCount; index++)
|
|
|
|
{
|
2023-10-13 21:18:39 +02:00
|
|
|
if (!this.findAndAddRandomPresetToLoot(globalDefaultPresets, itemTypeCounts, itemBlacklistArray, result))
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
index--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2023-04-23 00:41:04 +02:00
|
|
|
* Construct item limit record to hold max and current item count for each item type
|
2023-03-03 16:23:46 +01:00
|
|
|
* @param limits limits as defined in config
|
|
|
|
* @returns record, key: item tplId, value: current/max item count allowed
|
|
|
|
*/
|
2023-04-23 00:41:04 +02:00
|
|
|
protected initItemLimitCounter(limits: Record<string, number>): Record<string, ItemLimit>
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
2023-04-23 00:41:04 +02:00
|
|
|
const itemTypeCounts: Record<string, ItemLimit> = {};
|
|
|
|
for (const itemTypeId in limits)
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
2023-04-23 00:41:04 +02:00
|
|
|
itemTypeCounts[itemTypeId] = {
|
2023-03-03 16:23:46 +01:00
|
|
|
current: 0,
|
2023-11-13 17:05:05 +01:00
|
|
|
max: limits[itemTypeId],
|
2023-03-03 16:23:46 +01:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return itemTypeCounts;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Find a random item in items.json and add to result array
|
|
|
|
* @param items items to choose from
|
|
|
|
* @param itemTypeCounts item limit counts
|
|
|
|
* @param options item filters
|
|
|
|
* @param result array to add found item to
|
|
|
|
* @returns true if item was valid and added to pool
|
|
|
|
*/
|
|
|
|
protected findAndAddRandomItemToLoot(
|
|
|
|
items: [string, ITemplateItem][],
|
2023-11-13 17:05:05 +01:00
|
|
|
itemTypeCounts: Record<string, {current: number; max: number;}>,
|
2023-03-03 16:23:46 +01:00
|
|
|
options: LootRequest,
|
2023-11-13 17:05:05 +01:00
|
|
|
result: LootItem[],
|
|
|
|
): boolean
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
const randomItem = this.randomUtil.getArrayValue(items)[1];
|
|
|
|
|
|
|
|
const itemLimitCount = itemTypeCounts[randomItem._parent];
|
|
|
|
if (itemLimitCount && itemLimitCount.current > itemLimitCount.max)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const newLootItem: LootItem = {
|
|
|
|
id: this.hashUtil.generate(),
|
|
|
|
tpl: randomItem._id,
|
|
|
|
isPreset: false,
|
2023-11-13 17:05:05 +01:00
|
|
|
stackCount: 1,
|
2023-03-03 16:23:46 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
// Check if armor has level in allowed whitelist
|
2023-11-13 17:05:05 +01:00
|
|
|
if (
|
|
|
|
randomItem._parent === BaseClasses.ARMOR ||
|
|
|
|
randomItem._parent === BaseClasses.VEST
|
|
|
|
)
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
if (!options.armorLevelWhitelist.includes(Number(randomItem._props.armorClass)))
|
|
|
|
{
|
2023-11-13 17:05:05 +01:00
|
|
|
return false;
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Special case - handle items that need a stackcount > 1
|
|
|
|
if (randomItem._props.StackMaxSize > 1)
|
|
|
|
{
|
|
|
|
newLootItem.stackCount = this.getRandomisedStackCount(randomItem, options);
|
|
|
|
}
|
2023-11-13 17:05:05 +01:00
|
|
|
|
2023-03-03 16:23:46 +01:00
|
|
|
newLootItem.tpl = randomItem._id;
|
|
|
|
result.push(newLootItem);
|
|
|
|
|
|
|
|
if (itemLimitCount)
|
|
|
|
{
|
|
|
|
// Increment item count as it's in limit array
|
|
|
|
itemLimitCount.current++;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Item added okay
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a randomised stack count for an item between its StackMinRandom and StackMaxSize values
|
|
|
|
* @param item item to get stack count of
|
|
|
|
* @param options loot options
|
|
|
|
* @returns stack count
|
|
|
|
*/
|
|
|
|
protected getRandomisedStackCount(item: ITemplateItem, options: LootRequest): number
|
|
|
|
{
|
|
|
|
let min = item._props.StackMinRandom;
|
|
|
|
let max = item._props.StackMaxSize;
|
|
|
|
|
|
|
|
if (options.itemStackLimits[item._id])
|
|
|
|
{
|
|
|
|
min = options.itemStackLimits[item._id].min;
|
|
|
|
max = options.itemStackLimits[item._id].max;
|
|
|
|
}
|
|
|
|
|
|
|
|
return this.randomUtil.getInt(min, max);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Find a random item in items.json and add to result array
|
|
|
|
* @param globalDefaultPresets presets to choose from
|
|
|
|
* @param itemTypeCounts item limit counts
|
|
|
|
* @param itemBlacklist items to skip
|
|
|
|
* @param result array to add found preset to
|
|
|
|
* @returns true if preset was valid and added to pool
|
|
|
|
*/
|
|
|
|
protected findAndAddRandomPresetToLoot(
|
2023-07-18 16:44:14 +02:00
|
|
|
globalDefaultPresets: [string, IPreset][],
|
2023-11-13 17:05:05 +01:00
|
|
|
itemTypeCounts: Record<string, {current: number; max: number;}>,
|
2023-03-03 16:23:46 +01:00
|
|
|
itemBlacklist: string[],
|
2023-11-13 17:05:05 +01:00
|
|
|
result: LootItem[],
|
|
|
|
): boolean
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
// Choose random preset and get details from item.json using encyclopedia value (encyclopedia === tplId)
|
|
|
|
const randomPreset = this.randomUtil.getArrayValue(globalDefaultPresets)[1];
|
2023-08-05 15:50:05 +02:00
|
|
|
if (!randomPreset?._encyclopedia)
|
|
|
|
{
|
2023-10-10 13:03:20 +02:00
|
|
|
this.logger.debug(`Airdrop - preset with id: ${randomPreset._id} lacks encyclopedia property, skipping`);
|
2023-08-05 15:50:05 +02:00
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const itemDetails = this.itemHelper.getItem(randomPreset._encyclopedia);
|
|
|
|
if (!itemDetails[0])
|
|
|
|
{
|
2023-10-10 13:03:20 +02:00
|
|
|
this.logger.debug(`Airdrop - Unable to find preset with tpl: ${randomPreset._encyclopedia}, skipping`);
|
2023-08-05 15:50:05 +02:00
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
// Skip blacklisted items
|
|
|
|
if (itemBlacklist.includes(randomPreset._items[0]._tpl))
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Some custom mod items are lacking a parent property
|
2023-08-05 15:50:05 +02:00
|
|
|
if (!itemDetails[1]._parent)
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
2023-08-05 15:50:05 +02:00
|
|
|
this.logger.error(this.localisationService.getText("loot-item_missing_parentid", itemDetails[1]?._name));
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check picked preset hasn't exceeded spawn limit
|
2023-08-05 15:50:05 +02:00
|
|
|
const itemLimitCount = itemTypeCounts[itemDetails[1]._parent];
|
2023-03-03 16:23:46 +01:00
|
|
|
if (itemLimitCount && itemLimitCount.current > itemLimitCount.max)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const newLootItem: LootItem = {
|
|
|
|
tpl: randomPreset._items[0]._tpl,
|
|
|
|
isPreset: true,
|
2023-11-13 17:05:05 +01:00
|
|
|
stackCount: 1,
|
2023-03-03 16:23:46 +01:00
|
|
|
};
|
2023-11-13 17:05:05 +01:00
|
|
|
|
2023-03-03 16:23:46 +01:00
|
|
|
result.push(newLootItem);
|
|
|
|
|
|
|
|
if (itemLimitCount)
|
|
|
|
{
|
|
|
|
// increment item count as its in limit array
|
|
|
|
itemLimitCount.current++;
|
|
|
|
}
|
2023-11-13 17:05:05 +01:00
|
|
|
|
2023-03-03 16:23:46 +01:00
|
|
|
// item added okay
|
|
|
|
return true;
|
|
|
|
}
|
2023-06-20 17:07:05 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Sealed weapon containers have a weapon + associated mods inside them + assortment of other things (food/meds)
|
2023-06-20 18:19:53 +02:00
|
|
|
* @param containerSettings sealed weapon container settings
|
2023-06-20 17:07:05 +02:00
|
|
|
* @returns Array of items to add to player inventory
|
|
|
|
*/
|
2023-06-20 18:19:53 +02:00
|
|
|
public getSealedWeaponCaseLoot(containerSettings: ISealedAirdropContainerSettings): AddItem[]
|
2023-06-20 17:07:05 +02:00
|
|
|
{
|
|
|
|
const itemsToReturn: AddItem[] = [];
|
|
|
|
|
|
|
|
// choose a weapon to give to the player (weighted)
|
2023-11-13 17:05:05 +01:00
|
|
|
const chosenWeaponTpl = this.weightedRandomHelper.getWeightedValue<string>(
|
|
|
|
containerSettings.weaponRewardWeight,
|
|
|
|
);
|
2023-06-20 17:07:05 +02:00
|
|
|
const weaponDetailsDb = this.itemHelper.getItem(chosenWeaponTpl);
|
|
|
|
if (!weaponDetailsDb[0])
|
|
|
|
{
|
2023-11-13 17:05:05 +01:00
|
|
|
this.logger.error(
|
|
|
|
this.localisationService.getText("loot-non_item_picked_as_sealed_weapon_crate_reward", chosenWeaponTpl),
|
|
|
|
);
|
2023-06-20 17:07:05 +02:00
|
|
|
|
|
|
|
return itemsToReturn;
|
|
|
|
}
|
2023-11-13 17:05:05 +01:00
|
|
|
|
2023-06-20 17:07:05 +02:00
|
|
|
// Get weapon preset - default or choose a random one from all possible
|
2023-11-13 17:05:05 +01:00
|
|
|
let chosenWeaponPreset = containerSettings.defaultPresetsOnly ?
|
|
|
|
this.presetHelper.getDefaultPreset(chosenWeaponTpl) :
|
|
|
|
this.randomUtil.getArrayValue(this.presetHelper.getPresets(chosenWeaponTpl));
|
2023-06-20 17:07:05 +02:00
|
|
|
|
2023-07-23 12:27:04 +02:00
|
|
|
if (!chosenWeaponPreset)
|
|
|
|
{
|
|
|
|
this.logger.warning(`Default preset for weapon ${chosenWeaponTpl} not found, choosing random instead`);
|
|
|
|
chosenWeaponPreset = this.randomUtil.getArrayValue(this.presetHelper.getPresets(chosenWeaponTpl));
|
|
|
|
}
|
|
|
|
|
2023-06-20 17:07:05 +02:00
|
|
|
// Add preset to return object
|
|
|
|
itemsToReturn.push({
|
|
|
|
count: 1,
|
|
|
|
// eslint-disable-next-line @typescript-eslint/naming-convention
|
|
|
|
item_id: chosenWeaponPreset._id,
|
2023-11-13 17:05:05 +01:00
|
|
|
isPreset: true,
|
2023-06-20 17:07:05 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
// Get items related to chosen weapon
|
|
|
|
const linkedItemsToWeapon = this.ragfairLinkedItemService.getLinkedDbItems(chosenWeaponTpl);
|
2023-11-13 17:05:05 +01:00
|
|
|
itemsToReturn.push(
|
|
|
|
...this.getSealedContainerWeaponModRewards(containerSettings, linkedItemsToWeapon, chosenWeaponPreset),
|
|
|
|
);
|
2023-06-20 17:07:05 +02:00
|
|
|
|
|
|
|
// Handle non-weapon mod reward types
|
|
|
|
itemsToReturn.push(...this.getSealedContainerNonWeaponModRewards(containerSettings, weaponDetailsDb[1]));
|
|
|
|
|
|
|
|
return itemsToReturn;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get non-weapon mod rewards for a sealed container
|
|
|
|
* @param containerSettings Sealed weapon container settings
|
|
|
|
* @param weaponDetailsDb Details for the weapon to reward player
|
|
|
|
* @returns AddItem array
|
|
|
|
*/
|
2023-11-13 17:05:05 +01:00
|
|
|
protected getSealedContainerNonWeaponModRewards(
|
|
|
|
containerSettings: ISealedAirdropContainerSettings,
|
|
|
|
weaponDetailsDb: ITemplateItem,
|
|
|
|
): AddItem[]
|
2023-06-20 17:07:05 +02:00
|
|
|
{
|
|
|
|
const rewards: AddItem[] = [];
|
|
|
|
|
|
|
|
for (const rewardTypeId in containerSettings.rewardTypeLimits)
|
|
|
|
{
|
|
|
|
const settings = containerSettings.rewardTypeLimits[rewardTypeId];
|
|
|
|
const rewardCount = this.randomUtil.getInt(settings.min, settings.max);
|
|
|
|
|
|
|
|
if (rewardCount === 0)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Edge case - ammo boxes
|
|
|
|
if (rewardTypeId === BaseClasses.AMMO_BOX)
|
|
|
|
{
|
|
|
|
// Get ammoboxes from db
|
2023-11-13 17:05:05 +01:00
|
|
|
const ammoBoxesDetails = containerSettings.ammoBoxWhitelist.map((x) =>
|
2023-06-20 17:07:05 +02:00
|
|
|
{
|
|
|
|
const itemDetails = this.itemHelper.getItem(x);
|
|
|
|
return itemDetails[1];
|
|
|
|
});
|
2023-11-13 17:05:05 +01:00
|
|
|
|
2023-06-20 17:07:05 +02:00
|
|
|
// Need to find boxes that matches weapons caliber
|
|
|
|
const weaponCaliber = weaponDetailsDb._props.ammoCaliber;
|
2023-11-13 17:05:05 +01:00
|
|
|
const ammoBoxesMatchingCaliber = ammoBoxesDetails.filter((x) => x._props.ammoCaliber === weaponCaliber);
|
2023-06-20 17:07:05 +02:00
|
|
|
if (ammoBoxesMatchingCaliber.length === 0)
|
|
|
|
{
|
|
|
|
this.logger.debug(`No ammo box with caliber ${weaponCaliber} found, skipping`);
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// No need to add ammo to box, inventoryHelper.addItem() will handle it
|
|
|
|
const chosenAmmoBox = this.randomUtil.getArrayValue(ammoBoxesMatchingCaliber);
|
|
|
|
rewards.push({
|
|
|
|
count: rewardCount,
|
|
|
|
// eslint-disable-next-line @typescript-eslint/naming-convention
|
|
|
|
item_id: chosenAmmoBox._id,
|
2023-11-13 17:05:05 +01:00
|
|
|
isPreset: false,
|
2023-06-20 17:07:05 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get all items of the desired type + not quest items + not globally blacklisted
|
2023-10-12 12:00:04 +02:00
|
|
|
const rewardItemPool = Object.values(this.databaseServer.getTables().templates.items)
|
2023-11-13 17:05:05 +01:00
|
|
|
.filter((x) =>
|
|
|
|
x._parent === rewardTypeId &&
|
|
|
|
x._type.toLowerCase() === "item" &&
|
|
|
|
!this.itemFilterService.isItemBlacklisted(x._id) &&
|
2023-11-13 18:00:49 +01:00
|
|
|
(!(containerSettings.allowBossItems || this.itemFilterService.isBossItem(x._id))) &&
|
2023-11-13 17:05:05 +01:00
|
|
|
!x._props.QuestItem
|
|
|
|
);
|
2023-06-20 17:07:05 +02:00
|
|
|
|
2023-10-12 12:00:04 +02:00
|
|
|
if (rewardItemPool.length === 0)
|
2023-06-20 17:07:05 +02:00
|
|
|
{
|
|
|
|
this.logger.debug(`No items with base type of ${rewardTypeId} found, skipping`);
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (let index = 0; index < rewardCount; index++)
|
|
|
|
{
|
|
|
|
// choose a random item from pool
|
2023-10-12 12:00:04 +02:00
|
|
|
const chosenRewardItem = this.randomUtil.getArrayValue(rewardItemPool);
|
2023-11-13 17:05:05 +01:00
|
|
|
this.addOrIncrementItemToArray(chosenRewardItem._id, rewards);
|
2023-06-20 17:07:05 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return rewards;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Iterate over the container weaponModRewardLimits settings and create an array of weapon mods to reward player
|
|
|
|
* @param containerSettings Sealed weapon container settings
|
|
|
|
* @param linkedItemsToWeapon All items that can be attached/inserted into weapon
|
|
|
|
* @param chosenWeaponPreset The weapon preset given to player as reward
|
|
|
|
* @returns AddItem array
|
|
|
|
*/
|
2023-11-13 17:05:05 +01:00
|
|
|
protected getSealedContainerWeaponModRewards(
|
|
|
|
containerSettings: ISealedAirdropContainerSettings,
|
|
|
|
linkedItemsToWeapon: ITemplateItem[],
|
|
|
|
chosenWeaponPreset: IPreset,
|
|
|
|
): AddItem[]
|
2023-06-20 17:07:05 +02:00
|
|
|
{
|
|
|
|
const modRewards: AddItem[] = [];
|
|
|
|
for (const rewardTypeId in containerSettings.weaponModRewardLimits)
|
|
|
|
{
|
|
|
|
const settings = containerSettings.weaponModRewardLimits[rewardTypeId];
|
|
|
|
const rewardCount = this.randomUtil.getInt(settings.min, settings.max);
|
|
|
|
|
|
|
|
// Nothing to add, skip reward type
|
|
|
|
if (rewardCount === 0)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2023-11-13 17:05:05 +01:00
|
|
|
// Get items that fulfil reward type criteria from items that fit on gun
|
|
|
|
const relatedItems = linkedItemsToWeapon.filter((x) =>
|
|
|
|
x._parent === rewardTypeId && !this.itemFilterService.isItemBlacklisted(x._id)
|
|
|
|
);
|
2023-06-20 17:07:05 +02:00
|
|
|
if (!relatedItems || relatedItems.length === 0)
|
|
|
|
{
|
2023-11-13 17:05:05 +01:00
|
|
|
this.logger.debug(
|
|
|
|
`No items found to fulfil reward type ${rewardTypeId} for weapon: ${chosenWeaponPreset._name}, skipping type`,
|
|
|
|
);
|
2023-06-20 17:07:05 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find a random item of the desired type and add as reward
|
2023-11-13 17:05:05 +01:00
|
|
|
for (let index = 0; index < rewardCount; index++)
|
2023-06-20 17:07:05 +02:00
|
|
|
{
|
|
|
|
const chosenItem = this.randomUtil.drawRandomFromList(relatedItems);
|
|
|
|
this.addOrIncrementItemToArray(chosenItem[0]._id, modRewards);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return modRewards;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handle event-related loot containers - currently just the halloween jack-o-lanterns that give food rewards
|
2023-11-13 17:05:05 +01:00
|
|
|
* @param rewardContainerDetails
|
2023-06-20 17:07:05 +02:00
|
|
|
* @returns AddItem array
|
|
|
|
*/
|
|
|
|
public getRandomLootContainerLoot(rewardContainerDetails: RewardDetails): AddItem[]
|
|
|
|
{
|
|
|
|
const itemsToReturn: AddItem[] = [];
|
|
|
|
|
|
|
|
// Get random items and add to newItemRequest
|
|
|
|
for (let index = 0; index < rewardContainerDetails.rewardCount; index++)
|
|
|
|
{
|
|
|
|
// Pick random reward from pool, add to request object
|
2023-11-13 17:05:05 +01:00
|
|
|
const chosenRewardItemTpl = this.weightedRandomHelper.getWeightedValue<string>(
|
|
|
|
rewardContainerDetails.rewardTplPool,
|
|
|
|
);
|
2023-06-20 17:07:05 +02:00
|
|
|
this.addOrIncrementItemToArray(chosenRewardItemTpl, itemsToReturn);
|
|
|
|
}
|
|
|
|
|
|
|
|
return itemsToReturn;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A bug in inventoryHelper.addItem() means you cannot add the same item to the array twice with a count of 1, it causes duplication
|
|
|
|
* Default adds 1, or increments count
|
|
|
|
* @param itemTplToAdd items tpl we want to add to array
|
|
|
|
* @param resultsArray Array to add item tpl to
|
|
|
|
*/
|
|
|
|
protected addOrIncrementItemToArray(itemTplToAdd: string, resultsArray: AddItem[]): void
|
|
|
|
{
|
2023-11-13 17:05:05 +01:00
|
|
|
const existingItemIndex = resultsArray.findIndex((x) => x.item_id === itemTplToAdd);
|
2023-06-20 17:07:05 +02:00
|
|
|
if (existingItemIndex > -1)
|
|
|
|
{
|
|
|
|
// Exists in array already, increment count
|
|
|
|
resultsArray[existingItemIndex].count++;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// eslint-disable-next-line @typescript-eslint/naming-convention
|
|
|
|
resultsArray.push({item_id: itemTplToAdd, count: 1, isPreset: false});
|
|
|
|
}
|
|
|
|
}
|
2023-11-13 17:05:05 +01:00
|
|
|
}
|