Server/project/src/controllers/LocationController.ts

204 lines
8.0 KiB
TypeScript
Raw Normal View History

2023-03-03 16:23:46 +01:00
import { inject, injectable } from "tsyringe";
import { LocationGenerator } from "../generators/LocationGenerator";
import { LootGenerator } from "../generators/LootGenerator";
import { WeightedRandomHelper } from "../helpers/WeightedRandomHelper";
2023-03-03 16:23:46 +01:00
import { ILocation } from "../models/eft/common/ILocation";
import { ILocationBase } from "../models/eft/common/ILocationBase";
import {
ILocationsGenerateAllResponse
} from "../models/eft/common/ILocationsSourceDestinationBase";
import { ILooseLoot, SpawnpointTemplate } from "../models/eft/common/ILooseLoot";
import { IAirdropLootResult } from "../models/eft/location/IAirdropLootResult";
import { AirdropTypeEnum } from "../models/enums/AirdropType";
2023-03-03 16:23:46 +01:00
import { ConfigTypes } from "../models/enums/ConfigTypes";
import { IAirdropConfig } from "../models/spt/config/IAirdropConfig";
import { ILocations } from "../models/spt/server/ILocations";
import { LootRequest } from "../models/spt/services/LootRequest";
import { ILogger } from "../models/spt/utils/ILogger";
import { ConfigServer } from "../servers/ConfigServer";
import { DatabaseServer } from "../servers/DatabaseServer";
import { LocalisationService } from "../services/LocalisationService";
import { HashUtil } from "../utils/HashUtil";
import { JsonUtil } from "../utils/JsonUtil";
import { TimeUtil } from "../utils/TimeUtil";
@injectable()
export class LocationController
{
protected airdropConfig: IAirdropConfig;
constructor(
@inject("JsonUtil") protected jsonUtil: JsonUtil,
@inject("HashUtil") protected hashUtil: HashUtil,
@inject("WeightedRandomHelper") protected weightedRandomHelper: WeightedRandomHelper,
2023-03-03 16:23:46 +01:00
@inject("WinstonLogger") protected logger: ILogger,
@inject("LocationGenerator") protected locationGenerator: LocationGenerator,
@inject("LocalisationService") protected localisationService: LocalisationService,
@inject("LootGenerator") protected lootGenerator: LootGenerator,
@inject("DatabaseServer") protected databaseServer: DatabaseServer,
@inject("TimeUtil") protected timeUtil: TimeUtil,
@inject("ConfigServer") protected configServer: ConfigServer
)
{
this.airdropConfig = this.configServer.getConfig(ConfigTypes.AIRDROP);
}
2023-07-15 15:49:25 +02:00
/* */
/**
* Handle client/location/getLocalloot
* Get a location (map) with generated loot data
* @param location Map to generate loot for
* @returns ILocationBase
*/
2023-03-03 16:23:46 +01:00
public get(location: string): ILocationBase
{
const name = location.toLowerCase().replace(" ", "");
return this.generate(name);
}
/* generates a random location preset to use for local session */
private generate(name: string): ILocationBase
2023-03-03 16:23:46 +01:00
{
const location: ILocation = this.databaseServer.getTables().locations[name];
const output: ILocationBase = this.jsonUtil.clone(location.base);
2023-03-03 16:23:46 +01:00
// const ids = {};
output.UnixDateTime = this.timeUtil.getTimestamp();
// don't generate loot on hideout
if (name === "hideout")
{
return output;
}
const locationName = location.base.Name;
// Copy loot data
2023-03-03 16:23:46 +01:00
const staticWeapons = this.jsonUtil.clone(this.databaseServer.getTables().loot.staticContainers[locationName].staticWeapons);
const staticContainers = this.jsonUtil.clone(this.databaseServer.getTables().loot.staticContainers[locationName].staticContainers);
const staticForced = this.jsonUtil.clone(this.databaseServer.getTables().loot.staticContainers[locationName].staticForced);
const staticLootDist = this.jsonUtil.clone(this.databaseServer.getTables().loot.staticLoot);
const staticAmmoDist = this.jsonUtil.clone(this.databaseServer.getTables().loot.staticAmmo);
output.Loot = [];
// Mounted weapons
2023-03-03 16:23:46 +01:00
for (const mi of staticWeapons)
{
output.Loot.push(mi);
}
let staticContainerCount = 0;
2023-03-03 16:23:46 +01:00
// static loot
for (const staticContainer of staticContainers)
2023-03-03 16:23:46 +01:00
{
const container = this.locationGenerator.generateContainerLoot(staticContainer, staticForced, staticLootDist, staticAmmoDist, name);
2023-03-03 16:23:46 +01:00
output.Loot.push(container);
staticContainerCount++;
2023-03-03 16:23:46 +01:00
}
this.logger.success(this.localisationService.getText("location-containers_generated_success", staticContainerCount));
2023-03-03 16:23:46 +01:00
// dyanmic loot
const dynamicLootDist: ILooseLoot = this.jsonUtil.clone(location.looseLoot);
const dynamicLoot: SpawnpointTemplate[] = this.locationGenerator.generateDynamicLoot(dynamicLootDist, staticAmmoDist, name);
for (const dli of dynamicLoot)
{
output.Loot.push(dli);
}
// done generating
this.logger.success(this.localisationService.getText("location-dynamic_items_spawned_success", dynamicLoot.length));
this.logger.success(this.localisationService.getText("location-generated_success", name));
return output;
}
/**
* Handle client/locations
* Get all maps base location properties without loot data
* @param sessionId Players Id
* @returns ILocationsGenerateAllResponse
*/
// eslint-disable-next-line @typescript-eslint/no-unused-vars
public generateAll(sessionId: string): ILocationsGenerateAllResponse
2023-03-03 16:23:46 +01:00
{
const locationsFromDb = this.jsonUtil.clone(this.databaseServer.getTables().locations);
const locations: ILocations = {};
for (const mapName in locationsFromDb)
2023-03-03 16:23:46 +01:00
{
const mapBase = locationsFromDb[mapName]?.base;
if (!mapBase)
2023-03-03 16:23:46 +01:00
{
this.logger.debug(`Map: ${mapName} has no base json file, skipping generation`);
2023-03-03 16:23:46 +01:00
continue;
}
// Clear out loot array
mapBase.Loot = [];
// Add map base data to dictionary
locations[mapBase._Id] = mapBase;
2023-03-03 16:23:46 +01:00
}
return {
locations: locations,
paths: locationsFromDb.base.paths
};
2023-03-03 16:23:46 +01:00
}
/**
2023-07-15 15:49:25 +02:00
* Handle client/location/getAirdropLoot
2023-03-03 16:23:46 +01:00
* Get loot for an airdop container
* Generates it randomly based on config/airdrop.json values
* @returns Array of LootItem objects
2023-03-03 16:23:46 +01:00
*/
public getAirdropLoot(): IAirdropLootResult
2023-03-03 16:23:46 +01:00
{
const airdropType = this.chooseAirdropType();
this.logger.debug(`Chose ${airdropType} for airdrop loot`);
const airdropConfig = this.getAirdropLootConfigByType(airdropType);
return {dropType: airdropType, loot: this.lootGenerator.createRandomLoot(airdropConfig)};
}
/**
* Randomly pick a type of airdrop loot using weighted values from config
* @returns airdrop type value
*/
protected chooseAirdropType(): AirdropTypeEnum
{
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): LootRequest
{
let lootSettingsByType = this.airdropConfig.loot[airdropType];
if (!lootSettingsByType)
{
2023-07-19 12:00:34 +02:00
this.logger.error(this.localisationService.getText("location-unable_to_find_airdrop_drop_config_of_type", airdropType));
lootSettingsByType = this.airdropConfig.loot[AirdropTypeEnum.MIXED];
}
2023-03-03 16:23:46 +01:00
return {
presetCount: lootSettingsByType.presetCount,
itemCount: lootSettingsByType.itemCount,
2023-06-20 17:59:15 +02:00
weaponCrateCount: lootSettingsByType.weaponCrateCount,
itemBlacklist: lootSettingsByType.itemBlacklist,
itemTypeWhitelist: lootSettingsByType.itemTypeWhitelist,
itemLimits: lootSettingsByType.itemLimits,
itemStackLimits: lootSettingsByType.itemStackLimits,
armorLevelWhitelist: lootSettingsByType.armorLevelWhitelist
};
2023-03-03 16:23:46 +01:00
}
}