0
0
mirror of https://github.com/sp-tarkov/server.git synced 2025-02-13 09:50:43 -05:00
server/project/src/services/AirdropService.ts

181 lines
7.7 KiB
TypeScript
Raw Normal View History

2024-07-06 13:39:56 +01:00
import { LootGenerator } from "@spt/generators/LootGenerator";
import { ItemHelper } from "@spt/helpers/ItemHelper";
import { WeightedRandomHelper } from "@spt/helpers/WeightedRandomHelper";
import { IItem } from "@spt/models/eft/common/tables/IItem";
import { IGetAirdropLootRequest } from "@spt/models/eft/location/IGetAirdropLootRequest";
2024-07-06 13:39:56 +01:00
import { IGetAirdropLootResponse } from "@spt/models/eft/location/IGetAirdropLootResponse";
import { AirdropTypeEnum, SptAirdropTypeEnum } from "@spt/models/enums/AirdropType";
2024-07-06 13:39:56 +01:00
import { ConfigTypes } from "@spt/models/enums/ConfigTypes";
import { ItemTpl } from "@spt/models/enums/ItemTpl";
import { IAirdropConfig, IAirdropLoot } from "@spt/models/spt/config/IAirdropConfig";
2024-10-19 11:21:02 +01:00
import { IAirdropLootRequest, ILootRequest } from "@spt/models/spt/services/ILootRequest";
2024-07-06 13:39:56 +01:00
import { ILogger } from "@spt/models/spt/utils/ILogger";
import { ConfigServer } from "@spt/servers/ConfigServer";
import { DatabaseService } from "@spt/services/DatabaseService";
import { ItemFilterService } from "@spt/services/ItemFilterService";
import { LocalisationService } from "@spt/services/LocalisationService";
import { HashUtil } from "@spt/utils/HashUtil";
import { ICloner } from "@spt/utils/cloners/ICloner";
import { inject, injectable } from "tsyringe";
2024-07-06 13:39:56 +01:00
@injectable()
export class AirdropService {
2024-07-06 13:39:56 +01:00
protected airdropConfig: IAirdropConfig;
constructor(
@inject("PrimaryLogger") protected logger: ILogger,
@inject("HashUtil") protected hashUtil: HashUtil,
@inject("ItemHelper") protected itemHelper: ItemHelper,
@inject("WeightedRandomHelper") protected weightedRandomHelper: WeightedRandomHelper,
@inject("LocalisationService") protected localisationService: LocalisationService,
@inject("ItemFilterService") protected itemFilterService: ItemFilterService,
@inject("LootGenerator") protected lootGenerator: LootGenerator,
@inject("DatabaseService") protected databaseService: DatabaseService,
@inject("ConfigServer") protected configServer: ConfigServer,
@inject("PrimaryCloner") protected cloner: ICloner,
) {
2024-07-06 13:39:56 +01:00
this.airdropConfig = this.configServer.getConfig(ConfigTypes.AIRDROP);
}
public generateCustomAirdropLoot(request: IGetAirdropLootRequest): IGetAirdropLootResponse {
const customAirdropInformation = this.airdropConfig.customAirdropMapping[request.containerId];
if (!customAirdropInformation) {
this.logger.warning(
`Unable to find data for custom airdrop ${request.containerId}, returning random airdrop instead`,
);
return this.generateAirdropLoot();
}
return this.generateAirdropLoot(customAirdropInformation);
}
2024-07-06 13:39:56 +01:00
/**
* Handle client/location/getAirdropLoot
* Get loot for an airdrop container
* Generates it randomly based on config/airdrop.json values
2024-10-19 11:21:02 +01:00
* @param forcedAirdropType OPTIONAL - Desired airdrop type, randomised when not provided
2024-07-06 13:39:56 +01:00
* @returns Array of LootItem objects
*/
public generateAirdropLoot(forcedAirdropType = null): IGetAirdropLootResponse {
const airdropType = forcedAirdropType ? forcedAirdropType : this.chooseAirdropType();
2024-10-19 11:21:02 +01:00
this.logger.debug(`Chose: ${airdropType} for airdrop loot`);
2024-07-06 13:39:56 +01:00
// Common/weapon/etc
const airdropConfig = this.getAirdropLootConfigByType(airdropType);
// generate loot to put into airdrop crate
const crateLoot = airdropConfig.useForcedLoot
2024-10-19 11:21:02 +01:00
? this.lootGenerator.createForcedLoot(airdropConfig.forcedLoot)
: this.lootGenerator.createRandomLoot(airdropConfig);
2024-07-06 13:39:56 +01:00
// Create airdrop crate and add to result in first spot
const airdropCrateItem = this.getAirdropCrateItem(airdropType);
// Add crate to front of array
crateLoot.unshift(airdropCrateItem);
// Reparent loot items to crate we added above
for (const item of crateLoot) {
2024-07-23 17:30:20 +01:00
if (item._id === airdropCrateItem._id) {
2024-07-06 13:39:56 +01:00
// Crate itself, don't alter
continue;
}
// no parentId = root item, make item have create as parent
if (!item.parentId) {
2024-07-06 13:39:56 +01:00
item.parentId = airdropCrateItem._id;
item.slotId = "main";
}
}
return { icon: airdropConfig.icon, container: crateLoot };
2024-07-06 13:39:56 +01:00
}
/**
* Create a container create item based on passed in airdrop type
* @param airdropType What tpye of container: weapon/common etc
* @returns Item
*/
protected getAirdropCrateItem(airdropType: SptAirdropTypeEnum): IItem {
2024-07-06 13:39:56 +01:00
const airdropContainer = {
_id: this.hashUtil.generate(),
_tpl: "", // picked later
upd: {
SpawnedInSession: true,
StackObjectsCount: 1,
},
};
switch (airdropType) {
case SptAirdropTypeEnum.FOOD_MEDICAL:
2024-07-06 13:39:56 +01:00
airdropContainer._tpl = ItemTpl.LOOTCONTAINER_AIRDROP_MEDICAL_CRATE;
break;
case SptAirdropTypeEnum.SUPPLY:
2024-07-06 13:39:56 +01:00
airdropContainer._tpl = ItemTpl.LOOTCONTAINER_AIRDROP_SUPPLY_CRATE;
break;
case SptAirdropTypeEnum.WEAPON_ARMOR:
2024-07-06 13:39:56 +01:00
airdropContainer._tpl = ItemTpl.LOOTCONTAINER_AIRDROP_WEAPON_CRATE;
break;
case SptAirdropTypeEnum.COMMON:
airdropContainer._tpl = ItemTpl.LOOTCONTAINER_AIRDROP_COMMON_SUPPLY_CRATE;
break;
case SptAirdropTypeEnum.RADAR:
airdropContainer._tpl = ItemTpl.LOOTCONTAINER_AIRDROP_TECHNICAL_SUPPLY_CRATE_EVENT_1;
break;
2024-07-06 13:39:56 +01:00
default:
airdropContainer._tpl = ItemTpl.LOOTCONTAINER_AIRDROP_COMMON_SUPPLY_CRATE;
break;
}
return airdropContainer;
}
/**
* Randomly pick a type of airdrop loot using weighted values from config
* @returns airdrop type value
*/
protected chooseAirdropType(): SptAirdropTypeEnum {
2024-07-06 13:39:56 +01:00
const possibleAirdropTypes = this.airdropConfig.airdropTypeWeightings;
return this.weightedRandomHelper.getWeightedValue(possibleAirdropTypes);
}
/**
* Get the configuration for a specific type of airdrop
* @param airdropType Type of airdrop to get settings for
* @returns LootRequest
*/
protected getAirdropLootConfigByType(airdropType: AirdropTypeEnum): IAirdropLootRequest {
let lootSettingsByType: IAirdropLoot = this.airdropConfig.loot[airdropType];
if (!lootSettingsByType) {
2024-07-06 13:39:56 +01:00
this.logger.error(
this.localisationService.getText("location-unable_to_find_airdrop_drop_config_of_type", airdropType),
);
// Default to common
2024-07-06 13:39:56 +01:00
lootSettingsByType = this.airdropConfig.loot[AirdropTypeEnum.COMMON];
}
return {
icon: lootSettingsByType.icon,
2024-07-06 13:39:56 +01:00
weaponPresetCount: lootSettingsByType.weaponPresetCount,
armorPresetCount: lootSettingsByType.armorPresetCount,
itemCount: lootSettingsByType.itemCount,
weaponCrateCount: lootSettingsByType.weaponCrateCount,
itemBlacklist: [
...lootSettingsByType.itemBlacklist,
...this.itemFilterService.getItemRewardBlacklist(),
...this.itemFilterService.getBossItems(),
],
itemTypeWhitelist: lootSettingsByType.itemTypeWhitelist,
itemLimits: lootSettingsByType.itemLimits,
itemStackLimits: lootSettingsByType.itemStackLimits,
armorLevelWhitelist: lootSettingsByType.armorLevelWhitelist,
allowBossItems: lootSettingsByType.allowBossItems,
useForcedLoot: lootSettingsByType.useForcedLoot,
forcedLoot: lootSettingsByType.forcedLoot,
2024-07-06 13:39:56 +01:00
};
}
}