Server/project/src/generators/LocationGenerator.ts

1145 lines
46 KiB
TypeScript
Raw Normal View History

2023-03-03 15:23:46 +00:00
import { inject, injectable } from "tsyringe";
import { ContainerHelper } from "@spt/helpers/ContainerHelper";
import { ItemHelper } from "@spt/helpers/ItemHelper";
import { PresetHelper } from "@spt/helpers/PresetHelper";
import {
IContainerMinMax,
IStaticAmmoDetails,
IStaticContainer,
IStaticContainerData,
IStaticForcedProps,
IStaticLootDetails,
} from "@spt/models/eft/common/ILocation";
import { ILocationBase } from "@spt/models/eft/common/ILocationBase";
import { ILooseLoot, Spawnpoint, SpawnpointTemplate, SpawnpointsForced } from "@spt/models/eft/common/ILooseLoot";
import { Item } from "@spt/models/eft/common/tables/IItem";
import { BaseClasses } from "@spt/models/enums/BaseClasses";
import { ConfigTypes } from "@spt/models/enums/ConfigTypes";
import { Money } from "@spt/models/enums/Money";
import { ILocationConfig } from "@spt/models/spt/config/ILocationConfig";
import { ILogger } from "@spt/models/spt/utils/ILogger";
import { ConfigServer } from "@spt/servers/ConfigServer";
import { DatabaseService } from "@spt/services/DatabaseService";
import { LocalisationService } from "@spt/services/LocalisationService";
import { SeasonalEventService } from "@spt/services/SeasonalEventService";
import { ICloner } from "@spt/utils/cloners/ICloner";
import { MathUtil } from "@spt/utils/MathUtil";
import { ObjectId } from "@spt/utils/ObjectId";
import { ProbabilityObject, ProbabilityObjectArray, RandomUtil } from "@spt/utils/RandomUtil";
2023-03-03 15:23:46 +00:00
export interface IContainerItem
{
items: Item[]
width: number
height: number
2023-03-03 15:23:46 +00:00
}
export interface IContainerGroupCount
{
/** Containers this group has + probabilty to spawn */
containerIdsWithProbability: Record<string, number>
/** How many containers the map should spawn with this group id */
chosenCount: number
}
2023-03-03 15:23:46 +00:00
@injectable()
export class LocationGenerator
{
protected locationConfig: ILocationConfig;
constructor(
@inject("PrimaryLogger") protected logger: ILogger,
@inject("DatabaseService") protected databaseService: DatabaseService,
2023-03-03 15:23:46 +00:00
@inject("ObjectId") protected objectId: ObjectId,
@inject("RandomUtil") protected randomUtil: RandomUtil,
@inject("ItemHelper") protected itemHelper: ItemHelper,
@inject("MathUtil") protected mathUtil: MathUtil,
@inject("SeasonalEventService") protected seasonalEventService: SeasonalEventService,
@inject("ContainerHelper") protected containerHelper: ContainerHelper,
@inject("PresetHelper") protected presetHelper: PresetHelper,
@inject("LocalisationService") protected localisationService: LocalisationService,
@inject("ConfigServer") protected configServer: ConfigServer,
@inject("PrimaryCloner") protected cloner: ICloner,
2023-03-03 15:23:46 +00:00
)
{
this.locationConfig = this.configServer.getConfig(ConfigTypes.LOCATION);
}
/**
* Create an array of container objects with randomised loot
* @param locationBase Map base to generate containers for
* @param staticAmmoDist Static ammo distribution - database.loot.staticAmmo
* @returns Array of container objects
*/
public generateStaticContainers(
locationBase: ILocationBase,
staticAmmoDist: Record<string, IStaticAmmoDetails[]>,
): SpawnpointTemplate[]
{
let staticLootItemCount = 0;
const result: SpawnpointTemplate[] = [];
const locationId = locationBase.Id.toLowerCase();
const mapData = this.databaseService.getLocation(locationId);
const staticWeaponsOnMapClone = this.cloner.clone(mapData.staticContainers.staticWeapons);
if (!staticWeaponsOnMapClone)
{
2024-05-21 12:40:16 +01:00
this.logger.error(this.localisationService.getText("location-unable_to_find_static_weapon_for_map", locationBase.Name));
}
// Add mounted weapons to output loot
result.push(...(staticWeaponsOnMapClone ?? []));
const allStaticContainersOnMapClone = this.cloner.clone(mapData.staticContainers.staticContainers);
if (!allStaticContainersOnMapClone)
{
2024-05-21 12:40:16 +01:00
this.logger.error(this.localisationService.getText("location-unable_to_find_static_container_for_map", locationBase.Name));
}
const staticRandomisableContainersOnMap = this.getRandomisableContainersOnMap(allStaticContainersOnMapClone);
// Containers that MUST be added to map (quest containers etc)
const staticForcedOnMapClone = this.cloner.clone(mapData.staticContainers.staticForced);
if (!staticForcedOnMapClone)
{
2024-05-21 12:40:16 +01:00
this.logger.error(this.localisationService.getText("location-unable_to_find_forced_static_data_for_map", locationBase.Name));
}
// Keep track of static loot count
let staticContainerCount = 0;
// Find all 100% spawn containers
const staticLootDist = mapData.staticLoot;
const guaranteedContainers = this.getGuaranteedContainers(allStaticContainersOnMapClone);
staticContainerCount += guaranteedContainers.length;
// Add loot to guaranteed containers and add to result
for (const container of guaranteedContainers)
{
const containerWithLoot = this.addLootToContainer(
container,
staticForcedOnMapClone,
staticLootDist,
staticAmmoDist,
locationId,
);
result.push(containerWithLoot.template);
staticLootItemCount += containerWithLoot.template.Items.length;
}
this.logger.debug(`Added ${guaranteedContainers.length} guaranteed containers`);
// Randomisation is turned off globally or just turned off for this map
if (
!(
this.locationConfig.containerRandomisationSettings.enabled
&& this.locationConfig.containerRandomisationSettings.maps[locationId]
)
)
{
this.logger.debug(
`Container randomisation disabled, Adding ${staticRandomisableContainersOnMap.length} containers to ${locationBase.Name}`,
);
for (const container of staticRandomisableContainersOnMap)
{
const containerWithLoot = this.addLootToContainer(
container,
staticForcedOnMapClone,
staticLootDist,
staticAmmoDist,
locationId,
);
result.push(containerWithLoot.template);
staticLootItemCount += containerWithLoot.template.Items.length;
}
this.logger.success(`A total of ${staticLootItemCount} static items spawned`);
return result;
}
// Group containers by their groupId
if (!mapData.statics)
{
2024-05-21 14:28:52 +01:00
this.logger.warning(this.localisationService.getText("location-unable_to_generate_static_loot", locationId));
return result;
}
const mapping = this.getGroupIdToContainerMappings(mapData.statics, staticRandomisableContainersOnMap);
// For each of the container groups, choose from the pool of containers, hydrate container with loot and add to result array
for (const groupId in mapping)
{
const data = mapping[groupId];
// Count chosen was 0, skip
if (data.chosenCount === 0)
{
continue;
}
if (Object.keys(data.containerIdsWithProbability).length === 0)
{
this.logger.debug(
`Group: ${groupId} has no containers with < 100% spawn chance to choose from, skipping`,
);
continue;
}
// EDGE CASE: These are containers without a group and have a probability < 100%
if (groupId === "")
{
const containerIdsCopy = this.cloner.clone(data.containerIdsWithProbability);
// Roll each containers probability, if it passes, it gets added
data.containerIdsWithProbability = {};
for (const containerId in containerIdsCopy)
{
if (this.randomUtil.getChance100(containerIdsCopy[containerId] * 100))
{
data.containerIdsWithProbability[containerId] = containerIdsCopy[containerId];
}
}
// Set desired count to size of array (we want all containers chosen)
data.chosenCount = Object.keys(data.containerIdsWithProbability).length;
// EDGE CASE: chosen container count could be 0
if (data.chosenCount === 0)
{
continue;
}
}
// Pass possible containers into function to choose some
const chosenContainerIds = this.getContainersByProbabilty(groupId, data);
for (const chosenContainerId of chosenContainerIds)
{
// Look up container object from full list of containers on map
const containerObject = staticRandomisableContainersOnMap.find(
(staticContainer) => staticContainer.template.Id === chosenContainerId,
);
if (!containerObject)
{
this.logger.debug(
`Container: ${chosenContainerIds[chosenContainerId]} not found in staticRandomisableContainersOnMap, this is bad`,
);
continue;
}
// Add loot to container and push into result object
const containerWithLoot = this.addLootToContainer(
containerObject,
staticForcedOnMapClone,
staticLootDist,
staticAmmoDist,
locationId,
);
result.push(containerWithLoot.template);
staticContainerCount++;
staticLootItemCount += containerWithLoot.template.Items.length;
}
}
this.logger.success(`A total of ${staticLootItemCount} static items spawned`);
this.logger.success(
this.localisationService.getText("location-containers_generated_success", staticContainerCount),
);
return result;
}
/**
* Get containers with a non-100% chance to spawn OR are NOT on the container type randomistion blacklist
* @param staticContainers
* @returns IStaticContainerData array
*/
protected getRandomisableContainersOnMap(staticContainers: IStaticContainerData[]): IStaticContainerData[]
{
return staticContainers.filter(
(staticContainer) =>
staticContainer.probability !== 1
&& !staticContainer.template.IsAlwaysSpawn
&& !this.locationConfig.containerRandomisationSettings.containerTypesToNotRandomise.includes(
staticContainer.template.Items[0]._tpl,
),
);
}
/**
* Get containers with 100% spawn rate or have a type on the randomistion ignore list
* @param staticContainersOnMap
* @returns IStaticContainerData array
*/
protected getGuaranteedContainers(staticContainersOnMap: IStaticContainerData[]): IStaticContainerData[]
{
return staticContainersOnMap.filter(
(staticContainer) =>
staticContainer.probability === 1
|| staticContainer.template.IsAlwaysSpawn
|| this.locationConfig.containerRandomisationSettings.containerTypesToNotRandomise.includes(
staticContainer.template.Items[0]._tpl,
),
);
}
/**
* Choose a number of containers based on their probabilty value to fulfil the desired count in containerData.chosenCount
* @param groupId Name of the group the containers are being collected for
* @param containerData Containers and probability values for a groupId
* @returns List of chosen container Ids
*/
protected getContainersByProbabilty(groupId: string, containerData: IContainerGroupCount): string[]
{
const chosenContainerIds: string[] = [];
const containerIds = Object.keys(containerData.containerIdsWithProbability);
if (containerData.chosenCount > containerIds.length)
{
this.logger.debug(
`Group: ${groupId} wants ${containerData.chosenCount} containers but pool only has ${containerIds.length}, add what's available`,
);
return containerIds;
}
// Create probability array with all possible container ids in this group and their relataive probability of spawning
const containerDistribution = new ProbabilityObjectArray<string>(this.mathUtil, this.cloner);
for (const x of containerIds)
{
containerDistribution.push(new ProbabilityObject(x, containerData.containerIdsWithProbability[x]));
}
chosenContainerIds.push(...containerDistribution.draw(containerData.chosenCount));
return chosenContainerIds;
}
/**
* Get a mapping of each groupid and the containers in that group + count of containers to spawn on map
* @param containersGroups Container group values
* @returns dictionary keyed by groupId
*/
protected getGroupIdToContainerMappings(
staticContainerGroupData: IStaticContainer | Record<string, IContainerMinMax>,
staticContainersOnMap: IStaticContainerData[],
): Record<string, IContainerGroupCount>
{
// Create dictionary of all group ids and choose a count of containers the map will spawn of that group
const mapping: Record<string, IContainerGroupCount> = {};
for (const groupId in staticContainerGroupData.containersGroups)
{
const groupData = staticContainerGroupData.containersGroups[groupId];
if (!mapping[groupId])
{
mapping[groupId] = {
containerIdsWithProbability: {},
chosenCount: this.randomUtil.getInt(
Math.round(
groupData.minContainers
* this.locationConfig.containerRandomisationSettings.containerGroupMinSizeMultiplier,
),
Math.round(
groupData.maxContainers
* this.locationConfig.containerRandomisationSettings.containerGroupMaxSizeMultiplier,
),
),
};
}
}
// Add an empty group for containers without a group id but still have a < 100% chance to spawn
// Likely bad BSG data, will be fixed...eventually, example of the groupids: `NEED_TO_BE_FIXED1`,`NEED_TO_BE_FIXED_SE02`, `NEED_TO_BE_FIXED_NW_01`
mapping[""] = { containerIdsWithProbability: {}, chosenCount: -1 };
// Iterate over all containers and add to group keyed by groupId
// Containers without a group go into a group with empty key ""
for (const container of staticContainersOnMap)
{
const groupData = staticContainerGroupData.containers[container.template.Id];
if (!groupData)
{
2024-05-21 12:40:16 +01:00
this.logger.error(this.localisationService.getText("location-unable_to_find_container_in_statics_json", container.template.Id));
continue;
}
if (container.probability === 1)
{
this.logger.debug(
`Container ${container.template.Id} with group ${groupData.groupId} had 100% chance to spawn was picked as random container, skipping`,
);
2024-05-21 12:40:16 +01:00
continue;
}
mapping[groupData.groupId].containerIdsWithProbability[container.template.Id] = container.probability;
}
return mapping;
}
/**
* Choose loot to put into a static container based on weighting
* Handle forced items + seasonal item removal when not in season
* @param staticContainer The container itself we will add loot to
* @param staticForced Loot we need to force into the container
* @param staticLootDist staticLoot.json
* @param staticAmmoDist staticAmmo.json
* @param locationName Name of the map to generate static loot for
* @returns IStaticContainerProps
*/
protected addLootToContainer(
staticContainer: IStaticContainerData,
2023-03-03 15:23:46 +00:00
staticForced: IStaticForcedProps[],
staticLootDist: Record<string, IStaticLootDetails>,
staticAmmoDist: Record<string, IStaticAmmoDetails[]>,
locationName: string,
): IStaticContainerData
2023-03-03 15:23:46 +00:00
{
const containerClone = this.cloner.clone(staticContainer);
const containerTpl = containerClone.template.Items[0]._tpl;
// Create new unique parent id to prevent any collisions
2023-03-03 15:23:46 +00:00
const parentId = this.objectId.generate();
containerClone.template.Root = parentId;
containerClone.template.Items[0]._id = parentId;
2023-03-03 15:23:46 +00:00
2024-02-11 12:16:08 +00:00
const containerMap = this.getContainerMapping(containerTpl);
2023-03-03 15:23:46 +00:00
// Choose count of items to add to container
const itemCountToAdd = this.getWeightedCountOfContainerItems(containerTpl, staticLootDist, locationName);
2023-03-03 15:23:46 +00:00
// Get all possible loot items for container
const containerLootPool = this.getPossibleLootItemsForContainer(containerTpl, staticLootDist);
2023-03-03 15:23:46 +00:00
// Some containers need to have items forced into it (quest keys etc)
const tplsForced = staticForced
.filter((forcedStaticProp) => forcedStaticProp.containerId === containerClone.template.Id)
.map((x) => x.itemTpl);
2023-03-03 15:23:46 +00:00
// Draw random loot
// Money spawn more than once in container
let failedToFitCount = 0;
2023-03-03 15:23:46 +00:00
const locklist = [Money.ROUBLES, Money.DOLLARS, Money.EUROS];
// Choose items to add to container, factor in weighting + lock money down
2024-01-07 20:20:25 +00:00
// Filter out items picked that're already in the above `tplsForced` array
const chosenTpls = containerLootPool
.draw(itemCountToAdd, this.locationConfig.allowDuplicateItemsInStaticContainers, locklist)
.filter((tpl) => !tplsForced.includes(tpl));
// Add forced loot to chosen item pool
const tplsToAddToContainer = tplsForced.concat(chosenTpls);
for (const tplToAdd of tplsToAddToContainer)
2023-03-03 15:23:46 +00:00
{
const chosenItemWithChildren = this.createStaticLootItem(tplToAdd, staticAmmoDist, parentId);
const items = chosenItemWithChildren.items;
const width = chosenItemWithChildren.width;
const height = chosenItemWithChildren.height;
2023-03-03 15:23:46 +00:00
// look for open slot to put chosen item into
const result = this.containerHelper.findSlotForItem(containerMap, width, height);
2023-03-03 15:23:46 +00:00
if (!result.success)
{
if (failedToFitCount >= this.locationConfig.fitLootIntoContainerAttempts)
{
// x attempts to fit an item, container is probably full, stop trying to add more
break;
}
// Can't fit item, skip
failedToFitCount++;
continue;
2023-03-03 15:23:46 +00:00
}
this.containerHelper.fillContainerMapWithItem(
containerMap,
result.x,
result.y,
width,
height,
result.rotation,
);
const rotation = result.rotation ? 1 : 0;
2023-03-03 15:23:46 +00:00
items[0].slotId = "main";
items[0].location = { x: result.x, y: result.y, r: rotation };
2023-03-03 15:23:46 +00:00
// Add loot to container before returning
2023-03-03 15:23:46 +00:00
for (const item of items)
{
containerClone.template.Items.push(item);
2023-03-03 15:23:46 +00:00
}
}
return containerClone;
2023-03-03 15:23:46 +00:00
}
/**
* Get a 2d grid of a containers item slots
* @param containerTpl Tpl id of the container
* @returns number[][]
*/
protected getContainerMapping(containerTpl: string): number[][]
{
// Get template from db
const containerTemplate = this.itemHelper.getItem(containerTpl)[1];
// Get height/width
const height = containerTemplate._props.Grids[0]._props.cellsV;
const width = containerTemplate._props.Grids[0]._props.cellsH;
// Calcualte 2d array and return
return Array(height)
.fill(0)
.map(() => Array(width).fill(0));
}
/**
* Look up a containers itemcountDistribution data and choose an item count based on the found weights
* @param containerTypeId Container to get item count for
* @param staticLootDist staticLoot.json
* @param locationName Map name (to get per-map multiplier for from config)
* @returns item count
*/
protected getWeightedCountOfContainerItems(
containerTypeId: string,
staticLootDist: Record<string, IStaticLootDetails>,
locationName: string,
): number
{
// Create probability array to calcualte the total count of lootable items inside container
const itemCountArray = new ProbabilityObjectArray<number>(this.mathUtil, this.cloner);
const countDistribution = staticLootDist[containerTypeId]?.itemcountDistribution;
if (!countDistribution)
{
2024-05-24 16:42:42 +01:00
this.logger.warning(this.localisationService.getText("location-unable_to_find_count_distribution_for_container",
{
containerId: containerTypeId,
locationName: locationName,
}));
return 0;
}
for (const itemCountDistribution of countDistribution)
{
// Add each count of items into array
itemCountArray.push(
new ProbabilityObject(itemCountDistribution.count, itemCountDistribution.relativeProbability),
);
}
return Math.round(this.getStaticLootMultiplerForLocation(locationName) * itemCountArray.draw()[0]);
}
/**
* Get all possible loot items that can be placed into a container
* Do not add seasonal items if found + current date is inside seasonal event
* @param containerTypeId Contianer to get possible loot for
* @param staticLootDist staticLoot.json
* @returns ProbabilityObjectArray of item tpls + probabilty
*/
protected getPossibleLootItemsForContainer(
containerTypeId: string,
staticLootDist: Record<string, IStaticLootDetails>,
): ProbabilityObjectArray<string, number>
{
const seasonalEventActive = this.seasonalEventService.seasonalEventEnabled();
const seasonalItemTplBlacklist = this.seasonalEventService.getInactiveSeasonalEventItems();
const itemDistribution = new ProbabilityObjectArray<string>(this.mathUtil, this.cloner);
const itemContainerDistribution = staticLootDist[containerTypeId]?.itemDistribution;
if (!itemContainerDistribution)
{
2024-05-21 14:28:52 +01:00
this.logger.warning(this.localisationService.getText("location-missing_item_distribution_data", containerTypeId));
return itemDistribution;
}
for (const icd of itemContainerDistribution)
{
if (!seasonalEventActive && seasonalItemTplBlacklist.includes(icd.tpl))
{
// Skip seasonal event items if they're not enabled
continue;
}
itemDistribution.push(new ProbabilityObject(icd.tpl, icd.relativeProbability));
}
return itemDistribution;
}
2023-03-03 15:23:46 +00:00
protected getLooseLootMultiplerForLocation(location: string): number
{
return this.locationConfig.looseLootMultiplier[location];
}
protected getStaticLootMultiplerForLocation(location: string): number
{
return this.locationConfig.staticLootMultiplier[location];
}
/**
* Create array of loose + forced loot using probability system
* @param dynamicLootDist
* @param staticAmmoDist
2023-03-03 15:23:46 +00:00
* @param locationName Location to generate loot for
* @returns Array of spawn points with loot in them
*/
public generateDynamicLoot(
dynamicLootDist: ILooseLoot,
staticAmmoDist: Record<string, IStaticAmmoDetails[]>,
locationName: string,
): SpawnpointTemplate[]
2023-03-03 15:23:46 +00:00
{
const loot: SpawnpointTemplate[] = [];
const dynamicForcedSpawnPoints: SpawnpointsForced[] = [];
2023-03-03 15:23:46 +00:00
// Build the list of forced loot from both `spawnpointsForced` and any point marked `IsAlwaysSpawn`
dynamicForcedSpawnPoints.push(...dynamicLootDist.spawnpointsForced);
dynamicForcedSpawnPoints.push(...dynamicLootDist.spawnpoints.filter((point) => point.template.IsAlwaysSpawn));
// Add forced loot
this.addForcedLoot(loot, dynamicForcedSpawnPoints, locationName);
2023-03-03 15:23:46 +00:00
const allDynamicSpawnpoints = dynamicLootDist.spawnpoints;
// Draw from random distribution
const desiredSpawnpointCount = Math.round(
this.getLooseLootMultiplerForLocation(locationName)
* this.randomUtil.getNormallyDistributedRandomNumber(
dynamicLootDist.spawnpointCount.mean,
dynamicLootDist.spawnpointCount.std,
),
2023-03-03 15:23:46 +00:00
);
// Positions not in forced but have 100% chance to spawn
const guaranteedLoosePoints: Spawnpoint[] = [];
const blacklistedSpawnpoints = this.locationConfig.looseLootBlacklist[locationName];
const spawnpointArray = new ProbabilityObjectArray<string, Spawnpoint>(this.mathUtil, this.cloner);
for (const spawnpoint of allDynamicSpawnpoints)
2023-03-03 15:23:46 +00:00
{
// Point is blacklsited, skip
if (blacklistedSpawnpoints?.includes(spawnpoint.template.Id))
{
this.logger.debug(`Ignoring loose loot location: ${spawnpoint.template.Id}`);
continue;
}
// We've handled IsAlwaysSpawn above, so skip them
if (spawnpoint.template.IsAlwaysSpawn)
{
continue;
}
if (spawnpoint.probability === 1)
{
guaranteedLoosePoints.push(spawnpoint);
continue;
}
spawnpointArray.push(new ProbabilityObject(spawnpoint.template.Id, spawnpoint.probability, spawnpoint));
2023-03-03 15:23:46 +00:00
}
// Select a number of spawn points to add loot to
// Add ALL loose loot with 100% chance to pool
let chosenSpawnpoints: Spawnpoint[] = [...guaranteedLoosePoints];
2024-03-30 14:29:08 -04:00
const randomSpawnpointCount = desiredSpawnpointCount - chosenSpawnpoints.length;
// only draw random spawn points if needed
if (randomSpawnpointCount > 0 && spawnpointArray.length > 0)
2023-03-03 15:23:46 +00:00
{
// Add randomly chosen spawn points
for (const si of spawnpointArray.draw(randomSpawnpointCount, false))
{
chosenSpawnpoints.push(spawnpointArray.data(si));
}
2023-03-03 15:23:46 +00:00
}
// Filter out duplicate locationIds
chosenSpawnpoints = [
...new Map(chosenSpawnpoints.map((spawnPoint) => [spawnPoint.locationId, spawnPoint])).values(),
];
// Do we have enough items in pool to fulfill requirement
const tooManySpawnPointsRequested = desiredSpawnpointCount - chosenSpawnpoints.length > 0;
if (tooManySpawnPointsRequested)
2023-03-03 15:23:46 +00:00
{
this.logger.debug(
this.localisationService.getText("location-spawn_point_count_requested_vs_found", {
requested: desiredSpawnpointCount + guaranteedLoosePoints.length,
found: chosenSpawnpoints.length,
mapName: locationName,
}),
);
2023-03-03 15:23:46 +00:00
}
// Iterate over spawnpoints
2023-03-03 15:23:46 +00:00
const seasonalEventActive = this.seasonalEventService.seasonalEventEnabled();
const seasonalItemTplBlacklist = this.seasonalEventService.getInactiveSeasonalEventItems();
for (const spawnPoint of chosenSpawnpoints)
2023-03-03 15:23:46 +00:00
{
if (!spawnPoint.template)
{
this.logger.warning(
this.localisationService.getText("location-missing_dynamic_template", spawnPoint.locationId),
);
continue;
}
if (!spawnPoint.template.Items || spawnPoint.template.Items.length === 0)
{
this.logger.error(
this.localisationService.getText("location-spawnpoint_missing_items", spawnPoint.template.Id),
);
continue;
}
const itemArray = new ProbabilityObjectArray<string>(this.mathUtil, this.cloner);
2023-03-03 15:23:46 +00:00
for (const itemDist of spawnPoint.itemDistribution)
{
if (
!seasonalEventActive
&& seasonalItemTplBlacklist.includes(
spawnPoint.template.Items.find((item) => item._id === itemDist.composedKey.key)._tpl,
)
)
2023-03-03 15:23:46 +00:00
{
// Skip seasonal event items if they're not enabled
continue;
}
itemArray.push(new ProbabilityObject(itemDist.composedKey.key, itemDist.relativeProbability));
2023-03-03 15:23:46 +00:00
}
if (itemArray.length === 0)
{
2024-05-21 14:28:52 +01:00
this.logger.warning(this.localisationService.getText("location-loot_pool_is_empty_skipping", spawnPoint.template.Id));
continue;
}
2023-03-03 15:23:46 +00:00
// Draw a random item from spawn points possible items
const chosenComposedKey = itemArray.draw(1)[0];
const createItemResult = this.createDynamicLootItem(chosenComposedKey, spawnPoint, staticAmmoDist);
2023-03-03 15:23:46 +00:00
// Root id can change when generating a weapon
spawnPoint.template.Root = createItemResult.items[0]._id;
spawnPoint.template.Items = createItemResult.items;
loot.push(spawnPoint.template);
}
return loot;
}
/**
* Add forced spawn point loot into loot parameter array
* @param lootLocationTemplates array to add forced loot spawn locations to
* @param forcedSpawnPoints forced Forced loot locations that must be added
* @param locationName Name of map currently having force loot created for
2023-03-03 15:23:46 +00:00
*/
protected addForcedLoot(
lootLocationTemplates: SpawnpointTemplate[],
forcedSpawnPoints: SpawnpointsForced[],
locationName: string,
): void
2023-03-03 15:23:46 +00:00
{
const lootToForceSingleAmountOnMap = this.locationConfig.forcedLootSingleSpawnById[locationName];
if (lootToForceSingleAmountOnMap)
{
// Process loot items defined as requiring only 1 spawn position as they appear in multiple positions on the map
for (const itemTpl of lootToForceSingleAmountOnMap)
{
// Get all spawn positions for item tpl in forced loot array
const items = forcedSpawnPoints.filter(
(forcedSpawnPoint) => forcedSpawnPoint.template.Items[0]._tpl === itemTpl,
);
2023-03-03 15:23:46 +00:00
if (!items || items.length === 0)
{
this.logger.debug(
`Unable to adjust loot item ${itemTpl} as it does not exist inside ${locationName} forced loot.`,
);
2023-03-03 15:23:46 +00:00
continue;
}
// Create probability array of all spawn positions for this spawn id
const spawnpointArray = new ProbabilityObjectArray<string, SpawnpointsForced>(
this.mathUtil,
this.cloner,
);
2023-03-03 15:23:46 +00:00
for (const si of items)
{
// use locationId as template.Id is the same across all items
spawnpointArray.push(new ProbabilityObject(si.locationId, si.probability, si));
2023-03-03 15:23:46 +00:00
}
2023-03-03 15:23:46 +00:00
// Choose 1 out of all found spawn positions for spawn id and add to loot array
for (const spawnPointLocationId of spawnpointArray.draw(1, false))
{
const itemToAdd = items.find((item) => item.locationId === spawnPointLocationId);
2023-03-03 15:23:46 +00:00
const lootItem = itemToAdd.template;
lootItem.Root = this.objectId.generate();
lootItem.Items[0]._id = lootItem.Root;
lootLocationTemplates.push(lootItem);
2023-03-03 15:23:46 +00:00
}
}
}
const seasonalEventActive = this.seasonalEventService.seasonalEventEnabled();
const seasonalItemTplBlacklist = this.seasonalEventService.getInactiveSeasonalEventItems();
2023-03-03 15:23:46 +00:00
// Add remaining forced loot to array
for (const forcedLootLocation of forcedSpawnPoints)
2023-03-03 15:23:46 +00:00
{
const firstLootItemTpl = forcedLootLocation.template.Items[0]._tpl;
// Skip spawn positions processed already
if (lootToForceSingleAmountOnMap?.includes(firstLootItemTpl))
2023-03-03 15:23:46 +00:00
{
continue;
}
// Skip adding seasonal items when seasonal event is not active
if (!seasonalEventActive && seasonalItemTplBlacklist.includes(firstLootItemTpl))
2023-03-03 15:23:46 +00:00
{
continue;
}
const locationTemplateToAdd = forcedLootLocation.template;
// Ensure root id matches the first items id
locationTemplateToAdd.Root = this.objectId.generate();
locationTemplateToAdd.Items[0]._id = locationTemplateToAdd.Root;
// Push forced location into array as long as it doesnt exist already
const existingLocation = lootLocationTemplates.find(
(spawnPoint) => spawnPoint.Id === locationTemplateToAdd.Id,
);
if (!existingLocation)
{
lootLocationTemplates.push(locationTemplateToAdd);
}
else
{
this.logger.debug(
`Attempted to add a forced loot location with Id: ${locationTemplateToAdd.Id} to map ${locationName} that already has that id in use, skipping`,
);
}
2023-03-03 15:23:46 +00:00
}
}
/**
* Create array of item (with child items) and return
* @param chosenComposedKey Key we want to look up items for
* @param spawnPoint Dynamic spawn point item we want will be placed in
* @param staticAmmoDist ammo distributions
2023-03-03 15:23:46 +00:00
* @returns IContainerItem
*/
protected createDynamicLootItem(
chosenComposedKey: string,
spawnPoint: Spawnpoint,
staticAmmoDist: Record<string, IStaticAmmoDetails[]>,
): IContainerItem
2023-03-03 15:23:46 +00:00
{
const chosenItem = spawnPoint.template.Items.find((item) => item._id === chosenComposedKey);
2023-03-03 15:23:46 +00:00
const chosenTpl = chosenItem._tpl;
const itemTemplate = this.itemHelper.getItem(chosenTpl)[1];
2023-03-03 15:23:46 +00:00
// Item array to return
const itemWithMods: Item[] = [];
2023-03-03 15:23:46 +00:00
// Money/Ammo - don't rely on items in spawnPoint.template.Items so we can randomise it ourselves
2024-01-01 11:33:19 +00:00
if (this.itemHelper.isOfBaseclasses(chosenTpl, [BaseClasses.MONEY, BaseClasses.AMMO]))
2023-03-03 15:23:46 +00:00
{
const stackCount
= itemTemplate._props.StackMaxSize === 1
? 1
: this.randomUtil.getInt(itemTemplate._props.StackMinRandom, itemTemplate._props.StackMaxRandom);
itemWithMods.push({
_id: this.objectId.generate(),
_tpl: chosenTpl,
upd: { StackObjectsCount: stackCount },
});
2023-03-03 15:23:46 +00:00
}
else if (this.itemHelper.isOfBaseclass(chosenTpl, BaseClasses.AMMO_BOX))
{
2023-11-29 11:36:20 +00:00
// Fill with cartridges
const ammoBoxItem: Item[] = [{ _id: this.objectId.generate(), _tpl: chosenTpl }];
this.itemHelper.addCartridgesToAmmoBox(ammoBoxItem, itemTemplate);
itemWithMods.push(...ammoBoxItem);
}
else if (this.itemHelper.isOfBaseclass(chosenTpl, BaseClasses.MAGAZINE))
{
// Create array with just magazine
const magazineItem: Item[] = [{ _id: this.objectId.generate(), _tpl: chosenTpl }];
if (this.randomUtil.getChance100(this.locationConfig.staticMagazineLootHasAmmoChancePercent))
{
// Add randomised amount of cartridges
this.itemHelper.fillMagazineWithRandomCartridge(
magazineItem,
itemTemplate, // Magazine template
staticAmmoDist,
undefined,
this.locationConfig.minFillLooseMagazinePercent / 100,
);
}
itemWithMods.push(...magazineItem);
}
2023-03-03 15:23:46 +00:00
else
{
// Also used by armors to get child mods
2023-03-03 15:23:46 +00:00
// Get item + children and add into array we return
const itemWithChildren = this.itemHelper.findAndReturnChildrenAsItems(
spawnPoint.template.Items,
chosenItem._id,
);
2023-03-03 15:23:46 +00:00
// We need to reparent to ensure ids are unique
this.reparentItemAndChildren(itemWithChildren);
itemWithMods.push(...itemWithChildren);
}
// Get inventory size of item
const size = this.itemHelper.getItemSize(itemWithMods, itemWithMods[0]._id);
return { items: itemWithMods, width: size.width, height: size.height };
2023-03-03 15:23:46 +00:00
}
/**
* Replace the _id value for base item + all children items parentid value
* @param itemWithChildren Item with mods to update
* @param newId new id to add on chidren of base item
*/
protected reparentItemAndChildren(itemWithChildren: Item[], newId = this.objectId.generate()): void
{
// original id on base item
const oldId = itemWithChildren[0]._id;
// Update base item to use new id
itemWithChildren[0]._id = newId;
// Update all parentIds of items attached to base item to use new id
for (const item of itemWithChildren)
{
if (item.parentId === oldId)
{
item.parentId = newId;
}
}
}
/**
* Find an item in array by its _tpl, handle differently if chosenTpl is a weapon
* @param items Items array to search
* @param chosenTpl Tpl we want to get item with
* @returns Item object
*/
protected getItemInArray(items: Item[], chosenTpl: string): Item
{
if (this.itemHelper.isOfBaseclass(chosenTpl, BaseClasses.WEAPON))
{
return items.find((v) => v._tpl === chosenTpl && v.parentId === undefined);
2023-03-03 15:23:46 +00:00
}
return items.find((item) => item._tpl === chosenTpl);
2023-03-03 15:23:46 +00:00
}
// TODO: rewrite, BIG yikes
protected createStaticLootItem(
chosenTpl: string,
staticAmmoDist: Record<string, IStaticAmmoDetails[]>,
parentId: string = undefined,
): IContainerItem
2023-03-03 15:23:46 +00:00
{
const itemTemplate = this.itemHelper.getItem(chosenTpl)[1];
2023-03-03 15:23:46 +00:00
let width = itemTemplate._props.Width;
let height = itemTemplate._props.Height;
let items: Item[] = [{ _id: this.objectId.generate(), _tpl: chosenTpl }];
const rootItem = items[0];
2023-03-03 15:23:46 +00:00
// Use passed in parentId as override for new item
if (parentId)
{
rootItem.parentId = parentId;
2023-03-03 15:23:46 +00:00
}
if (
this.itemHelper.isOfBaseclass(chosenTpl, BaseClasses.MONEY)
|| this.itemHelper.isOfBaseclass(chosenTpl, BaseClasses.AMMO)
)
2023-03-03 15:23:46 +00:00
{
// Edge case - some ammos e.g. flares or M406 grenades shouldn't be stacked
const stackCount
= itemTemplate._props.StackMaxSize === 1
? 1
: this.randomUtil.getInt(itemTemplate._props.StackMinRandom, itemTemplate._props.StackMaxRandom);
rootItem.upd = { StackObjectsCount: stackCount };
2023-03-03 15:23:46 +00:00
}
// No spawn point, use default template
else if (this.itemHelper.isOfBaseclass(chosenTpl, BaseClasses.WEAPON))
2023-03-03 15:23:46 +00:00
{
let children: Item[] = [];
const defaultPreset = this.cloner.clone(this.presetHelper.getDefaultPreset(chosenTpl));
if (defaultPreset?._items)
2023-03-03 15:23:46 +00:00
{
try
{
children = this.itemHelper.reparentItemAndChildren(defaultPreset._items[0], defaultPreset._items);
2023-03-03 15:23:46 +00:00
}
catch (error)
{
// this item already broke it once without being reproducible tpl = "5839a40f24597726f856b511"; AKS-74UB Default
// 5ea03f7400685063ec28bfa8 // ppsh default
// 5ba26383d4351e00334c93d9 //mp7_devgru
this.logger.warning(
this.localisationService.getText("location-preset_not_found", {
tpl: chosenTpl,
defaultId: defaultPreset._id,
defaultName: defaultPreset._name,
parentId: parentId,
}),
);
2023-03-03 15:23:46 +00:00
throw error;
}
}
else
{
// RSP30 (62178be9d0050232da3485d9/624c0b3340357b5f566e8766/6217726288ed9f0845317459) doesnt have any default presets and kills this code below as it has no chidren to reparent
this.logger.debug(`createItem() No preset found for weapon: ${chosenTpl}`);
2023-03-03 15:23:46 +00:00
}
const rootItem = items[0];
if (!rootItem)
{
this.logger.error(
this.localisationService.getText("location-missing_root_item", {
tpl: chosenTpl,
parentId: parentId,
}),
);
2023-03-03 15:23:46 +00:00
throw new Error(this.localisationService.getText("location-critical_error_see_log"));
}
try
{
if (children?.length > 0)
{
items = this.itemHelper.reparentItemAndChildren(rootItem, children);
}
2023-03-03 15:23:46 +00:00
}
catch (error)
{
this.logger.error(
this.localisationService.getText("location-unable_to_reparent_item", {
tpl: chosenTpl,
parentId: parentId,
}),
);
2023-03-03 15:23:46 +00:00
throw error;
}
// Here we should use generalized BotGenerators functions e.g. fillExistingMagazines in the future since
// it can handle revolver ammo (it's not restructured to be used here yet.)
// General: Make a WeaponController for Ragfair preset stuff and the generating weapons and ammo stuff from
// BotGenerator
const magazine = items.filter((item) => item.slotId === "mod_magazine")[0];
2023-03-03 15:23:46 +00:00
// some weapon presets come without magazine; only fill the mag if it exists
if (magazine)
2023-03-03 15:23:46 +00:00
{
const magTemplate = this.itemHelper.getItem(magazine._tpl)[1];
const weaponTemplate = this.itemHelper.getItem(chosenTpl)[1];
// Create array with just magazine
const defaultWeapon = this.itemHelper.getItem(rootItem._tpl)[1];
const magazineWithCartridges = [magazine];
this.itemHelper.fillMagazineWithRandomCartridge(
magazineWithCartridges,
magTemplate,
staticAmmoDist,
weaponTemplate._props.ammoCaliber,
0.25,
defaultWeapon._props.defAmmo,
defaultWeapon,
);
// Replace existing magazine with above array
items.splice(items.indexOf(magazine), 1, ...magazineWithCartridges);
2023-03-03 15:23:46 +00:00
}
const size = this.itemHelper.getItemSize(items, rootItem._id);
width = size.width;
height = size.height;
}
// No spawnpoint to fall back on, generate manually
else if (this.itemHelper.isOfBaseclass(chosenTpl, BaseClasses.AMMO_BOX))
2023-03-03 15:23:46 +00:00
{
this.itemHelper.addCartridgesToAmmoBox(items, itemTemplate);
2023-03-03 15:23:46 +00:00
}
else if (this.itemHelper.isOfBaseclass(chosenTpl, BaseClasses.MAGAZINE))
2023-03-03 15:23:46 +00:00
{
if (this.randomUtil.getChance100(this.locationConfig.magazineLootHasAmmoChancePercent))
{
// Create array with just magazine
const magazineWithCartridges = [rootItem];
this.itemHelper.fillMagazineWithRandomCartridge(
magazineWithCartridges,
itemTemplate,
staticAmmoDist,
undefined,
this.locationConfig.minFillStaticMagazinePercent / 100,
);
// Replace existing magazine with above array
items.splice(items.indexOf(rootItem), 1, ...magazineWithCartridges);
}
2023-03-03 15:23:46 +00:00
}
else if (this.itemHelper.armorItemCanHoldMods(chosenTpl))
{
const defaultPreset = this.presetHelper.getDefaultPreset(chosenTpl);
if (defaultPreset)
{
const presetAndMods: Item[] = this.itemHelper.replaceIDs(defaultPreset._items);
this.itemHelper.remapRootItemId(presetAndMods);
// Use original items parentId otherwise item doesnt get added to container correctly
presetAndMods[0].parentId = rootItem.parentId;
items = presetAndMods;
}
else
{
// We make base item above, at start of function, no need to do it here
if (itemTemplate._props.Slots?.length > 0)
{
items = this.itemHelper.addChildSlotItems(
items,
itemTemplate,
this.locationConfig.equipmentLootSettings.modSpawnChancePercent,
);
}
}
}
return { items: items, width: width, height: height };
}
}