1410 lines
62 KiB
TypeScript
1410 lines
62 KiB
TypeScript
import { inject, injectable } from "tsyringe";
|
|
|
|
import { HandbookHelper } from "@spt-aki/helpers/HandbookHelper";
|
|
import { ItemHelper } from "@spt-aki/helpers/ItemHelper";
|
|
import { PresetHelper } from "@spt-aki/helpers/PresetHelper";
|
|
import { ProfileHelper } from "@spt-aki/helpers/ProfileHelper";
|
|
import { RagfairServerHelper } from "@spt-aki/helpers/RagfairServerHelper";
|
|
import { RepeatableQuestHelper } from "@spt-aki/helpers/RepeatableQuestHelper";
|
|
import { IPreset } from "@spt-aki/models/eft/common/IGlobals";
|
|
import { Exit, ILocationBase } from "@spt-aki/models/eft/common/ILocationBase";
|
|
import { TraderInfo } from "@spt-aki/models/eft/common/tables/IBotBase";
|
|
import { Item } from "@spt-aki/models/eft/common/tables/IItem";
|
|
import {
|
|
IQuestCondition,
|
|
IQuestConditionCounterCondition,
|
|
IQuestReward,
|
|
IQuestRewards,
|
|
} from "@spt-aki/models/eft/common/tables/IQuest";
|
|
import { IRepeatableQuest } from "@spt-aki/models/eft/common/tables/IRepeatableQuests";
|
|
import { ITemplateItem } from "@spt-aki/models/eft/common/tables/ITemplateItem";
|
|
import { BaseClasses } from "@spt-aki/models/enums/BaseClasses";
|
|
import { ConfigTypes } from "@spt-aki/models/enums/ConfigTypes";
|
|
import { Money } from "@spt-aki/models/enums/Money";
|
|
import { QuestRewardType } from "@spt-aki/models/enums/QuestRewardType";
|
|
import { Traders } from "@spt-aki/models/enums/Traders";
|
|
import {
|
|
IBaseQuestConfig,
|
|
IBossInfo,
|
|
IEliminationConfig,
|
|
IQuestConfig,
|
|
IRepeatableQuestConfig,
|
|
} from "@spt-aki/models/spt/config/IQuestConfig";
|
|
import { IQuestTypePool } from "@spt-aki/models/spt/repeatable/IQuestTypePool";
|
|
import { ExhaustableArray } from "@spt-aki/models/spt/server/ExhaustableArray";
|
|
import { ILogger } from "@spt-aki/models/spt/utils/ILogger";
|
|
import { EventOutputHolder } from "@spt-aki/routers/EventOutputHolder";
|
|
import { ConfigServer } from "@spt-aki/servers/ConfigServer";
|
|
import { DatabaseServer } from "@spt-aki/servers/DatabaseServer";
|
|
import { ItemFilterService } from "@spt-aki/services/ItemFilterService";
|
|
import { LocalisationService } from "@spt-aki/services/LocalisationService";
|
|
import { PaymentService } from "@spt-aki/services/PaymentService";
|
|
import { ProfileFixerService } from "@spt-aki/services/ProfileFixerService";
|
|
import { SeasonalEventService } from "@spt-aki/services/SeasonalEventService";
|
|
import { HttpResponseUtil } from "@spt-aki/utils/HttpResponseUtil";
|
|
import { JsonUtil } from "@spt-aki/utils/JsonUtil";
|
|
import { MathUtil } from "@spt-aki/utils/MathUtil";
|
|
import { ObjectId } from "@spt-aki/utils/ObjectId";
|
|
import { ProbabilityObjectArray, RandomUtil } from "@spt-aki/utils/RandomUtil";
|
|
import { TimeUtil } from "@spt-aki/utils/TimeUtil";
|
|
|
|
@injectable()
|
|
export class RepeatableQuestGenerator
|
|
{
|
|
protected questConfig: IQuestConfig;
|
|
|
|
constructor(
|
|
@inject("TimeUtil") protected timeUtil: TimeUtil,
|
|
@inject("WinstonLogger") protected logger: ILogger,
|
|
@inject("RandomUtil") protected randomUtil: RandomUtil,
|
|
@inject("HttpResponseUtil") protected httpResponse: HttpResponseUtil,
|
|
@inject("MathUtil") protected mathUtil: MathUtil,
|
|
@inject("JsonUtil") protected jsonUtil: JsonUtil,
|
|
@inject("DatabaseServer") protected databaseServer: DatabaseServer,
|
|
@inject("ItemHelper") protected itemHelper: ItemHelper,
|
|
@inject("PresetHelper") protected presetHelper: PresetHelper,
|
|
@inject("ProfileHelper") protected profileHelper: ProfileHelper,
|
|
@inject("ProfileFixerService") protected profileFixerService: ProfileFixerService,
|
|
@inject("HandbookHelper") protected handbookHelper: HandbookHelper,
|
|
@inject("RagfairServerHelper") protected ragfairServerHelper: RagfairServerHelper,
|
|
@inject("EventOutputHolder") protected eventOutputHolder: EventOutputHolder,
|
|
@inject("LocalisationService") protected localisationService: LocalisationService,
|
|
@inject("PaymentService") protected paymentService: PaymentService,
|
|
@inject("ObjectId") protected objectId: ObjectId,
|
|
@inject("ItemFilterService") protected itemFilterService: ItemFilterService,
|
|
@inject("RepeatableQuestHelper") protected repeatableQuestHelper: RepeatableQuestHelper,
|
|
@inject("SeasonalEventService") protected seasonalEventService: SeasonalEventService,
|
|
@inject("ConfigServer") protected configServer: ConfigServer,
|
|
)
|
|
{
|
|
this.questConfig = this.configServer.getConfig(ConfigTypes.QUEST);
|
|
}
|
|
|
|
/**
|
|
* This method is called by /GetClientRepeatableQuests/ and creates one element of quest type format (see assets/database/templates/repeatableQuests.json).
|
|
* It randomly draws a quest type (currently Elimination, Completion or Exploration) as well as a trader who is providing the quest
|
|
* @param pmcLevel Player's level for requested items and reward generation
|
|
* @param pmcTraderInfo Players traper standing/rep levels
|
|
* @param questTypePool Possible quest types pool
|
|
* @param repeatableConfig Repeatable quest config
|
|
* @returns IRepeatableQuest
|
|
*/
|
|
public generateRepeatableQuest(
|
|
pmcLevel: number,
|
|
pmcTraderInfo: Record<string, TraderInfo>,
|
|
questTypePool: IQuestTypePool,
|
|
repeatableConfig: IRepeatableQuestConfig,
|
|
): IRepeatableQuest
|
|
{
|
|
const questType = this.randomUtil.drawRandomFromList<string>(questTypePool.types)[0];
|
|
|
|
// get traders from whitelist and filter by quest type availability
|
|
let traders = repeatableConfig.traderWhitelist.filter((x) => x.questTypes.includes(questType)).map((x) =>
|
|
x.traderId
|
|
);
|
|
// filter out locked traders
|
|
traders = traders.filter((x) => pmcTraderInfo[x].unlocked);
|
|
const traderId = this.randomUtil.drawRandomFromList(traders)[0];
|
|
|
|
switch (questType)
|
|
{
|
|
case "Elimination":
|
|
return this.generateEliminationQuest(pmcLevel, traderId, questTypePool, repeatableConfig);
|
|
case "Completion":
|
|
return this.generateCompletionQuest(pmcLevel, traderId, repeatableConfig);
|
|
case "Exploration":
|
|
return this.generateExplorationQuest(pmcLevel, traderId, questTypePool, repeatableConfig);
|
|
case "Pickup":
|
|
return this.generatePickupQuest(pmcLevel, traderId, questTypePool, repeatableConfig);
|
|
default:
|
|
throw new Error(`Unknown mission type ${questType}. Should never be here!`);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Generate a randomised Elimination quest
|
|
* @param pmcLevel Player's level for requested items and reward generation
|
|
* @param traderId Trader from which the quest will be provided
|
|
* @param questTypePool Pools for quests (used to avoid redundant quests)
|
|
* @param repeatableConfig The configuration for the repeatably kind (daily, weekly) as configured in QuestConfig for the requestd quest
|
|
* @returns Object of quest type format for "Elimination" (see assets/database/templates/repeatableQuests.json)
|
|
*/
|
|
protected generateEliminationQuest(
|
|
pmcLevel: number,
|
|
traderId: string,
|
|
questTypePool: IQuestTypePool,
|
|
repeatableConfig: IRepeatableQuestConfig,
|
|
): IRepeatableQuest
|
|
{
|
|
const eliminationConfig = this.repeatableQuestHelper.getEliminationConfigByPmcLevel(pmcLevel, repeatableConfig);
|
|
const locationsConfig = repeatableConfig.locations;
|
|
let targetsConfig = this.repeatableQuestHelper.probabilityObjectArray(eliminationConfig.targets);
|
|
const bodypartsConfig = this.repeatableQuestHelper.probabilityObjectArray(eliminationConfig.bodyParts);
|
|
let weaponCategoryRequirementConfig = this.repeatableQuestHelper.probabilityObjectArray(
|
|
eliminationConfig.weaponCategoryRequirements,
|
|
);
|
|
const weaponRequirementConfig = this.repeatableQuestHelper.probabilityObjectArray(
|
|
eliminationConfig.weaponRequirements,
|
|
);
|
|
|
|
// the difficulty of the quest varies in difficulty depending on the condition
|
|
// possible conditions are
|
|
// - amount of npcs to kill
|
|
// - type of npc to kill (scav, boss, pmc)
|
|
// - with hit to what body part they should be killed
|
|
// - from what distance they should be killed
|
|
// a random combination of listed conditions can be required
|
|
// possible conditions elements and their relative probability can be defined in QuestConfig.js
|
|
// We use ProbabilityObjectArray to draw by relative probability. e.g. for targets:
|
|
// "targets": {
|
|
// "Savage": 7,
|
|
// "AnyPmc": 2,
|
|
// "bossBully": 0.5
|
|
// }
|
|
// higher is more likely. We define the difficulty to be the inverse of the relative probability.
|
|
|
|
// We want to generate a reward which is scaled by the difficulty of this mission. To get a upper bound with which we scale
|
|
// the actual difficulty we calculate the minimum and maximum difficulty (max being the sum of max of each condition type
|
|
// times the number of kills we have to perform):
|
|
|
|
// the minumum difficulty is the difficulty for the most probable (= easiest target) with no additional conditions
|
|
const minDifficulty = 1 / targetsConfig.maxProbability(); // min difficulty is lowest amount of scavs without any constraints
|
|
|
|
// Target on bodyPart max. difficulty is that of the least probable element
|
|
const maxTargetDifficulty = 1 / targetsConfig.minProbability();
|
|
const maxBodyPartsDifficulty = eliminationConfig.minKills / bodypartsConfig.minProbability();
|
|
|
|
// maxDistDifficulty is defined by 2, this could be a tuning parameter if we don't like the reward generation
|
|
const maxDistDifficulty = 2;
|
|
|
|
const maxKillDifficulty = eliminationConfig.maxKills;
|
|
|
|
function difficultyWeighing(
|
|
target: number,
|
|
bodyPart: number,
|
|
dist: number,
|
|
kill: number,
|
|
weaponRequirement: number,
|
|
): number
|
|
{
|
|
return Math.sqrt(Math.sqrt(target) + bodyPart + dist + weaponRequirement) * kill;
|
|
}
|
|
|
|
targetsConfig = targetsConfig.filter((x) =>
|
|
Object.keys(questTypePool.pool.Elimination.targets).includes(x.key)
|
|
);
|
|
if (targetsConfig.length === 0 || targetsConfig.every((x) => x.data.isBoss))
|
|
{
|
|
// There are no more targets left for elimination; delete it as a possible quest type
|
|
// also if only bosses are left we need to leave otherwise it's a guaranteed boss elimination
|
|
// -> then it would not be a quest with low probability anymore
|
|
questTypePool.types = questTypePool.types.filter((t) => t !== "Elimination");
|
|
return null;
|
|
}
|
|
|
|
const targetKey = targetsConfig.draw()[0];
|
|
const targetDifficulty = 1 / targetsConfig.probability(targetKey);
|
|
|
|
let locations: string[] = questTypePool.pool.Elimination.targets[targetKey].locations;
|
|
|
|
// we use any as location if "any" is in the pool and we do not hit the specific location random
|
|
// we use any also if the random condition is not met in case only "any" was in the pool
|
|
let locationKey = "any";
|
|
if (
|
|
locations.includes("any")
|
|
&& (eliminationConfig.specificLocationProb < Math.random() || locations.length <= 1)
|
|
)
|
|
{
|
|
locationKey = "any";
|
|
delete questTypePool.pool.Elimination.targets[targetKey];
|
|
}
|
|
else
|
|
{
|
|
locations = locations.filter((l) => l !== "any");
|
|
if (locations.length > 0)
|
|
{
|
|
locationKey = this.randomUtil.drawRandomFromList<string>(locations)[0];
|
|
questTypePool.pool.Elimination.targets[targetKey].locations = locations.filter((l) =>
|
|
l !== locationKey
|
|
);
|
|
if (questTypePool.pool.Elimination.targets[targetKey].locations.length === 0)
|
|
{
|
|
delete questTypePool.pool.Elimination.targets[targetKey];
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// never should reach this if everything works out
|
|
this.logger.debug("Ecountered issue when creating Elimination quest. Please report.");
|
|
}
|
|
}
|
|
|
|
// draw the target body part and calculate the difficulty factor
|
|
let bodyPartsToClient = null;
|
|
let bodyPartDifficulty = 0;
|
|
if (eliminationConfig.bodyPartProb > Math.random())
|
|
{
|
|
// if we add a bodyPart condition, we draw randomly one or two parts
|
|
// each bodyPart of the BODYPARTS ProbabilityObjectArray includes the string(s) which need to be presented to the client in ProbabilityObjectArray.data
|
|
// e.g. we draw "Arms" from the probability array but must present ["LeftArm", "RightArm"] to the client
|
|
bodyPartsToClient = [];
|
|
const bodyParts = bodypartsConfig.draw(this.randomUtil.randInt(1, 3), false);
|
|
let probability = 0;
|
|
for (const bi of bodyParts)
|
|
{
|
|
// more than one part lead to an "OR" condition hence more parts reduce the difficulty
|
|
probability += bodypartsConfig.probability(bi);
|
|
for (const biClient of bodypartsConfig.data(bi))
|
|
{
|
|
bodyPartsToClient.push(biClient);
|
|
}
|
|
}
|
|
bodyPartDifficulty = 1 / probability;
|
|
}
|
|
|
|
// Draw a distance condition
|
|
let distance = null;
|
|
let distanceDifficulty = 0;
|
|
let isDistanceRequirementAllowed = !eliminationConfig.distLocationBlacklist.includes(locationKey);
|
|
|
|
if (targetsConfig.data(targetKey).isBoss)
|
|
{
|
|
// Get all boss spawn information
|
|
const bossSpawns = Object.values(this.databaseServer.getTables().locations).filter((x) =>
|
|
"base" in x && "Id" in x.base
|
|
).map((x) => ({ Id: x.base.Id, BossSpawn: x.base.BossLocationSpawn }));
|
|
// filter for the current boss to spawn on map
|
|
const thisBossSpawns = bossSpawns.map((x) => ({
|
|
Id: x.Id,
|
|
BossSpawn: x.BossSpawn.filter((e) => e.BossName === targetKey),
|
|
})).filter((x) => x.BossSpawn.length > 0);
|
|
// remove blacklisted locations
|
|
const allowedSpawns = thisBossSpawns.filter((x) => !eliminationConfig.distLocationBlacklist.includes(x.Id));
|
|
// if the boss spawns on nom-blacklisted locations and the current location is allowed we can generate a distance kill requirement
|
|
isDistanceRequirementAllowed = isDistanceRequirementAllowed && (allowedSpawns.length > 0);
|
|
}
|
|
|
|
if (eliminationConfig.distProb > Math.random() && isDistanceRequirementAllowed)
|
|
{
|
|
// Random distance with lower values more likely; simple distribution for starters...
|
|
distance = Math.floor(
|
|
Math.abs(Math.random() - Math.random()) * (1 + eliminationConfig.maxDist - eliminationConfig.minDist)
|
|
+ eliminationConfig.minDist,
|
|
);
|
|
distance = Math.ceil(distance / 5) * 5;
|
|
distanceDifficulty = maxDistDifficulty * distance / eliminationConfig.maxDist;
|
|
}
|
|
|
|
let allowedWeaponsCategory: string = undefined;
|
|
if (eliminationConfig.weaponCategoryRequirementProb > Math.random())
|
|
{
|
|
// Filter out close range weapons from far distance requirement
|
|
if (distance > 50)
|
|
{
|
|
weaponCategoryRequirementConfig = weaponCategoryRequirementConfig.filter((category) =>
|
|
["Shotgun", "Pistol"].includes(category.key)
|
|
);
|
|
}
|
|
else if (distance < 20)
|
|
{ // Filter out far range weapons from close distance requirement
|
|
weaponCategoryRequirementConfig = weaponCategoryRequirementConfig.filter((category) =>
|
|
["MarksmanRifle", "DMR"].includes(category.key)
|
|
);
|
|
}
|
|
|
|
// Pick a weighted weapon category
|
|
const weaponRequirement = weaponCategoryRequirementConfig.draw(1, false);
|
|
|
|
// Get the hideout id value stored in the .data array
|
|
allowedWeaponsCategory = weaponCategoryRequirementConfig.data(weaponRequirement[0])[0];
|
|
}
|
|
|
|
// Only allow a specific weapon requirement if a weapon category was not chosen
|
|
let allowedWeapon: string = undefined;
|
|
if (!allowedWeaponsCategory && eliminationConfig.weaponRequirementProb > Math.random())
|
|
{
|
|
const weaponRequirement = weaponRequirementConfig.draw(1, false);
|
|
const allowedWeaponsCategory = weaponRequirementConfig.data(weaponRequirement[0])[0];
|
|
const allowedWeapons = this.itemHelper.getItemTplsOfBaseType(allowedWeaponsCategory);
|
|
allowedWeapon = this.randomUtil.getArrayValue(allowedWeapons);
|
|
}
|
|
|
|
// Draw how many npm kills are required
|
|
const desiredKillCount = this.getEliminationKillCount(targetKey, targetsConfig, eliminationConfig);
|
|
const killDifficulty = desiredKillCount;
|
|
|
|
// not perfectly happy here; we give difficulty = 1 to the quest reward generation when we have the most diffucult mission
|
|
// e.g. killing reshala 5 times from a distance of 200m with a headshot.
|
|
const maxDifficulty = difficultyWeighing(1, 1, 1, 1, 1);
|
|
const curDifficulty = difficultyWeighing(
|
|
targetDifficulty / maxTargetDifficulty,
|
|
bodyPartDifficulty / maxBodyPartsDifficulty,
|
|
distanceDifficulty / maxDistDifficulty,
|
|
killDifficulty / maxKillDifficulty,
|
|
(allowedWeaponsCategory || allowedWeapon) ? 1 : 0,
|
|
);
|
|
|
|
// Aforementioned issue makes it a bit crazy since now all easier quests give significantly lower rewards than Completion / Exploration
|
|
// I therefore moved the mapping a bit up (from 0.2...1 to 0.5...2) so that normal difficulty still gives good reward and having the
|
|
// crazy maximum difficulty will lead to a higher difficulty reward gain factor than 1
|
|
const difficulty = this.mathUtil.mapToRange(curDifficulty, minDifficulty, maxDifficulty, 0.5, 2);
|
|
|
|
const quest = this.generateRepeatableTemplate("Elimination", traderId, repeatableConfig.side);
|
|
|
|
// ASSUMPTION: All fence quests are for scavs
|
|
if (traderId === Traders.FENCE)
|
|
{
|
|
quest.side = "Scav";
|
|
}
|
|
|
|
const availableForFinishCondition = quest.conditions.AvailableForFinish[0];
|
|
availableForFinishCondition.counter.id = this.objectId.generate();
|
|
availableForFinishCondition.counter.conditions = [];
|
|
|
|
// Only add specific location condition if specific map selected
|
|
if (locationKey !== "any")
|
|
{
|
|
availableForFinishCondition.counter.conditions.push(
|
|
this.generateEliminationLocation(locationsConfig[locationKey]),
|
|
);
|
|
}
|
|
availableForFinishCondition.counter.conditions.push(
|
|
this.generateEliminationCondition(
|
|
targetKey,
|
|
bodyPartsToClient,
|
|
distance,
|
|
allowedWeapon,
|
|
allowedWeaponsCategory,
|
|
),
|
|
);
|
|
availableForFinishCondition.value = desiredKillCount;
|
|
availableForFinishCondition.id = this.objectId.generate();
|
|
quest.location = this.getQuestLocationByMapId(locationKey);
|
|
|
|
quest.rewards = this.generateReward(
|
|
pmcLevel,
|
|
Math.min(difficulty, 1),
|
|
traderId,
|
|
repeatableConfig,
|
|
eliminationConfig,
|
|
);
|
|
|
|
return quest;
|
|
}
|
|
|
|
/**
|
|
* Get a number of kills neded to complete elimination quest
|
|
* @param targetKey Target type desired e.g. anyPmc/bossBully/Savage
|
|
* @param targetsConfig Config
|
|
* @param eliminationConfig Config
|
|
* @returns Number of AI to kill
|
|
*/
|
|
protected getEliminationKillCount(
|
|
targetKey: string,
|
|
targetsConfig: ProbabilityObjectArray<string, IBossInfo>,
|
|
eliminationConfig: IEliminationConfig,
|
|
): number
|
|
{
|
|
if (targetsConfig.data(targetKey).isBoss)
|
|
{
|
|
return this.randomUtil.randInt(eliminationConfig.minBossKills, eliminationConfig.maxBossKills + 1);
|
|
}
|
|
|
|
if (targetsConfig.data(targetKey).isPmc)
|
|
{
|
|
return this.randomUtil.randInt(eliminationConfig.minPmcKills, eliminationConfig.maxPmcKills + 1);
|
|
}
|
|
|
|
return this.randomUtil.randInt(eliminationConfig.minKills, eliminationConfig.maxKills + 1);
|
|
}
|
|
|
|
/**
|
|
* A repeatable quest, besides some more or less static components, exists of reward and condition (see assets/database/templates/repeatableQuests.json)
|
|
* This is a helper method for GenerateEliminationQuest to create a location condition.
|
|
*
|
|
* @param {string} location the location on which to fulfill the elimination quest
|
|
* @returns {IEliminationCondition} object of "Elimination"-location-subcondition
|
|
*/
|
|
protected generateEliminationLocation(location: string[]): IQuestConditionCounterCondition
|
|
{
|
|
const propsObject: IQuestConditionCounterCondition = {
|
|
id: this.objectId.generate(),
|
|
dynamicLocale: true,
|
|
target: location,
|
|
conditionType: "Location",
|
|
};
|
|
|
|
return propsObject;
|
|
}
|
|
|
|
/**
|
|
* Create kill condition for an elimination quest
|
|
* @param target Bot type target of elimination quest e.g. "AnyPmc", "Savage"
|
|
* @param targetedBodyParts Body parts player must hit
|
|
* @param distance Distance from which to kill (currently only >= supported
|
|
* @param allowedWeapon What weapon must be used - undefined = any
|
|
* @param allowedWeaponCategory What category of weapon must be used - undefined = any
|
|
* @returns IEliminationCondition object
|
|
*/
|
|
protected generateEliminationCondition(
|
|
target: string,
|
|
targetedBodyParts: string[],
|
|
distance: number,
|
|
allowedWeapon: string,
|
|
allowedWeaponCategory: string,
|
|
): IQuestConditionCounterCondition
|
|
{
|
|
const killConditionProps: IQuestConditionCounterCondition = {
|
|
target: target,
|
|
value: 1,
|
|
id: this.objectId.generate(),
|
|
dynamicLocale: true,
|
|
conditionType: "Kills",
|
|
};
|
|
|
|
if (target.startsWith("boss"))
|
|
{
|
|
killConditionProps.target = "Savage";
|
|
killConditionProps.savageRole = [target];
|
|
}
|
|
|
|
// Has specific body part hit condition
|
|
if (targetedBodyParts)
|
|
{
|
|
killConditionProps.bodyPart = targetedBodyParts;
|
|
}
|
|
|
|
// Dont allow distance + melee requirement
|
|
if (distance && allowedWeaponCategory !== "5b5f7a0886f77409407a7f96")
|
|
{
|
|
killConditionProps.distance = { compareMethod: ">=", value: distance };
|
|
}
|
|
|
|
// Has specific weapon requirement
|
|
if (allowedWeapon)
|
|
{
|
|
killConditionProps.weapon = [allowedWeapon];
|
|
}
|
|
|
|
// Has specific weapon category requirement
|
|
if (allowedWeaponCategory?.length > 0)
|
|
{
|
|
// TODO - fix - does weaponCategories exist?
|
|
// killConditionProps.weaponCategories = [allowedWeaponCategory];
|
|
}
|
|
|
|
return killConditionProps;
|
|
}
|
|
|
|
/**
|
|
* Generates a valid Completion quest
|
|
*
|
|
* @param {integer} pmcLevel player's level for requested items and reward generation
|
|
* @param {string} traderId trader from which the quest will be provided
|
|
* @param {object} repeatableConfig The configuration for the repeatably kind (daily, weekly) as configured in QuestConfig for the requestd quest
|
|
* @returns {object} object of quest type format for "Completion" (see assets/database/templates/repeatableQuests.json)
|
|
*/
|
|
protected generateCompletionQuest(
|
|
pmcLevel: number,
|
|
traderId: string,
|
|
repeatableConfig: IRepeatableQuestConfig,
|
|
): IRepeatableQuest
|
|
{
|
|
const completionConfig = repeatableConfig.questConfig.Completion;
|
|
const levelsConfig = repeatableConfig.rewardScaling.levels;
|
|
const roublesConfig = repeatableConfig.rewardScaling.roubles;
|
|
|
|
const distinctItemsToRetrieveCount = this.randomUtil.getInt(1, completionConfig.uniqueItemCount);
|
|
|
|
const quest = this.generateRepeatableTemplate("Completion", traderId, repeatableConfig.side);
|
|
|
|
// Filter the items.json items to items the player must retrieve to complete quest: shouldn't be a quest item or "non-existant"
|
|
const possibleItemsToRetrievePool = this.getRewardableItems(repeatableConfig, traderId);
|
|
|
|
// Be fair, don't let the items be more expensive than the reward
|
|
let roublesBudget = Math.floor(
|
|
this.mathUtil.interp1(pmcLevel, levelsConfig, roublesConfig) * this.randomUtil.getFloat(0.5, 1),
|
|
);
|
|
roublesBudget = Math.max(roublesBudget, 5000);
|
|
let itemSelection = possibleItemsToRetrievePool.filter((x) =>
|
|
this.itemHelper.getItemPrice(x[0]) < roublesBudget
|
|
);
|
|
|
|
// We also have the option to use whitelist and/or blacklist which is defined in repeatableQuests.json as
|
|
// [{"minPlayerLevel": 1, "itemIds": ["id1",...]}, {"minPlayerLevel": 15, "itemIds": ["id3",...]}]
|
|
if (repeatableConfig.questConfig.Completion.useWhitelist)
|
|
{
|
|
const itemWhitelist =
|
|
this.databaseServer.getTables().templates.repeatableQuests.data.Completion.itemsWhitelist;
|
|
|
|
// Filter and concatenate the arrays according to current player level
|
|
const itemIdsWhitelisted = itemWhitelist.filter((p) => p.minPlayerLevel <= pmcLevel).reduce(
|
|
(a, p) => a.concat(p.itemIds),
|
|
[],
|
|
);
|
|
itemSelection = itemSelection.filter((x) =>
|
|
{
|
|
// Whitelist can contain item tpls and item base type ids
|
|
return (itemIdsWhitelisted.some((v) => this.itemHelper.isOfBaseclass(x[0], v))
|
|
|| itemIdsWhitelisted.includes(x[0]));
|
|
});
|
|
// check if items are missing
|
|
// const flatList = itemSelection.reduce((a, il) => a.concat(il[0]), []);
|
|
// const missing = itemIdsWhitelisted.filter(l => !flatList.includes(l));
|
|
}
|
|
|
|
if (repeatableConfig.questConfig.Completion.useBlacklist)
|
|
{
|
|
const itemBlacklist =
|
|
this.databaseServer.getTables().templates.repeatableQuests.data.Completion.itemsBlacklist;
|
|
// we filter and concatenate the arrays according to current player level
|
|
const itemIdsBlacklisted = itemBlacklist.filter((p) => p.minPlayerLevel <= pmcLevel).reduce(
|
|
(a, p) => a.concat(p.itemIds),
|
|
[],
|
|
);
|
|
itemSelection = itemSelection.filter((x) =>
|
|
{
|
|
return itemIdsBlacklisted.every((v) => !this.itemHelper.isOfBaseclass(x[0], v))
|
|
|| !itemIdsBlacklisted.includes(x[0]);
|
|
});
|
|
}
|
|
|
|
if (itemSelection.length === 0)
|
|
{
|
|
this.logger.error(
|
|
this.localisationService.getText(
|
|
"repeatable-completion_quest_whitelist_too_small_or_blacklist_too_restrictive",
|
|
),
|
|
);
|
|
|
|
return null;
|
|
}
|
|
|
|
// Draw items to ask player to retrieve
|
|
let isAmmo = 0;
|
|
const randomNumbersUsed = [];
|
|
for (let i = 0; i < distinctItemsToRetrieveCount; i++)
|
|
{
|
|
let randomNumber = this.randomUtil.randInt(itemSelection.length);
|
|
while (randomNumbersUsed.includes(randomNumber) && randomNumbersUsed.length !== itemSelection.length)
|
|
{
|
|
randomNumber = this.randomUtil.randInt(itemSelection.length);
|
|
}
|
|
|
|
randomNumbersUsed.push(randomNumber);
|
|
|
|
const itemSelected = itemSelection[randomNumber];
|
|
const itemUnitPrice = this.itemHelper.getItemPrice(itemSelected[0]);
|
|
let minValue = completionConfig.minRequestedAmount;
|
|
let maxValue = completionConfig.maxRequestedAmount;
|
|
if (this.itemHelper.isOfBaseclass(itemSelected[0], BaseClasses.AMMO))
|
|
{
|
|
// Prevent multiple ammo requirements from being picked, stop after 6 attempts
|
|
if (isAmmo > 0 && isAmmo < 6)
|
|
{
|
|
isAmmo++;
|
|
i--;
|
|
continue;
|
|
}
|
|
isAmmo++;
|
|
minValue = completionConfig.minRequestedBulletAmount;
|
|
maxValue = completionConfig.maxRequestedBulletAmount;
|
|
}
|
|
let value = minValue;
|
|
|
|
// get the value range within budget
|
|
maxValue = Math.min(maxValue, Math.floor(roublesBudget / itemUnitPrice));
|
|
if (maxValue > minValue)
|
|
{
|
|
// if it doesn't blow the budget we have for the request, draw a random amount of the selected
|
|
// item type to be requested
|
|
value = this.randomUtil.randInt(minValue, maxValue + 1);
|
|
}
|
|
roublesBudget -= value * itemUnitPrice;
|
|
|
|
// push a CompletionCondition with the item and the amount of the item
|
|
quest.conditions.AvailableForFinish.push(this.generateCompletionAvailableForFinish(itemSelected[0], value));
|
|
|
|
if (roublesBudget > 0)
|
|
{
|
|
// reduce the list possible items to fulfill the new budget constraint
|
|
itemSelection = itemSelection.filter((x) => this.itemHelper.getItemPrice(x[0]) < roublesBudget);
|
|
if (itemSelection.length === 0)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
quest.rewards = this.generateReward(pmcLevel, 1, traderId, repeatableConfig, completionConfig);
|
|
|
|
return quest;
|
|
}
|
|
|
|
/**
|
|
* A repeatable quest, besides some more or less static components, exists of reward and condition (see assets/database/templates/repeatableQuests.json)
|
|
* This is a helper method for GenerateCompletionQuest to create a completion condition (of which a completion quest theoretically can have many)
|
|
*
|
|
* @param {string} itemTpl id of the item to request
|
|
* @param {integer} value amount of items of this specific type to request
|
|
* @returns {object} object of "Completion"-condition
|
|
*/
|
|
protected generateCompletionAvailableForFinish(itemTpl: string, value: number): IQuestCondition
|
|
{
|
|
let minDurability = 0;
|
|
let onlyFoundInRaid = true;
|
|
if (
|
|
this.itemHelper.isOfBaseclass(itemTpl, BaseClasses.WEAPON)
|
|
|| this.itemHelper.isOfBaseclass(itemTpl, BaseClasses.ARMOR)
|
|
)
|
|
{
|
|
minDurability = this.randomUtil.getArrayValue([60, 80]);
|
|
}
|
|
|
|
// By default all collected items must be FiR, except dog tags
|
|
if (
|
|
this.itemHelper.isOfBaseclass(itemTpl, BaseClasses.DOG_TAG_USEC)
|
|
|| this.itemHelper.isOfBaseclass(itemTpl, BaseClasses.DOG_TAG_BEAR)
|
|
)
|
|
{
|
|
onlyFoundInRaid = false;
|
|
}
|
|
|
|
return {
|
|
id: this.objectId.generate(),
|
|
parentId: "",
|
|
dynamicLocale: true,
|
|
index: 0,
|
|
visibilityConditions: [],
|
|
target: [itemTpl],
|
|
value: value,
|
|
minDurability: minDurability,
|
|
maxDurability: 100,
|
|
dogtagLevel: 0,
|
|
onlyFoundInRaid: onlyFoundInRaid,
|
|
conditionType: "HandoverItem",
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Generates a valid Exploration quest
|
|
*
|
|
* @param {integer} pmcLevel player's level for reward generation
|
|
* @param {string} traderId trader from which the quest will be provided
|
|
* @param {object} questTypePool Pools for quests (used to avoid redundant quests)
|
|
* @param {object} repeatableConfig The configuration for the repeatably kind (daily, weekly) as configured in QuestConfig for the requestd quest
|
|
* @returns {object} object of quest type format for "Exploration" (see assets/database/templates/repeatableQuests.json)
|
|
*/
|
|
protected generateExplorationQuest(
|
|
pmcLevel: number,
|
|
traderId: string,
|
|
questTypePool: IQuestTypePool,
|
|
repeatableConfig: IRepeatableQuestConfig,
|
|
): IRepeatableQuest
|
|
{
|
|
const explorationConfig = repeatableConfig.questConfig.Exploration;
|
|
const requiresSpecificExtract =
|
|
Math.random() < repeatableConfig.questConfig.Exploration.specificExits.probability;
|
|
|
|
if (Object.keys(questTypePool.pool.Exploration.locations).length === 0)
|
|
{
|
|
// there are no more locations left for exploration; delete it as a possible quest type
|
|
questTypePool.types = questTypePool.types.filter((t) => t !== "Exploration");
|
|
return null;
|
|
}
|
|
|
|
// If location drawn is factory, it's possible to either get factory4_day and factory4_night or only one
|
|
// of the both
|
|
const locationKey: string = this.randomUtil.drawRandomFromDict(questTypePool.pool.Exploration.locations)[0];
|
|
const locationTarget = questTypePool.pool.Exploration.locations[locationKey];
|
|
|
|
// Remove the location from the available pool
|
|
delete questTypePool.pool.Exploration.locations[locationKey];
|
|
|
|
// Different max extract count when specific extract needed
|
|
const exitTimesMax = requiresSpecificExtract
|
|
? explorationConfig.maxExtractsWithSpecificExit
|
|
: explorationConfig.maxExtracts + 1;
|
|
const numExtracts = this.randomUtil.randInt(1, exitTimesMax);
|
|
|
|
const quest = this.generateRepeatableTemplate("Exploration", traderId, repeatableConfig.side);
|
|
|
|
const exitStatusCondition: IQuestConditionCounterCondition = {
|
|
conditionType: "ExitStatus",
|
|
id: this.objectId.generate(),
|
|
dynamicLocale: true,
|
|
status: ["Survived"],
|
|
};
|
|
const locationCondition: IQuestConditionCounterCondition = {
|
|
conditionType: "Location",
|
|
id: this.objectId.generate(),
|
|
dynamicLocale: true,
|
|
target: locationTarget,
|
|
};
|
|
|
|
quest.conditions.AvailableForFinish[0].counter.id = this.objectId.generate();
|
|
quest.conditions.AvailableForFinish[0].counter.conditions = [exitStatusCondition, locationCondition];
|
|
quest.conditions.AvailableForFinish[0].value = numExtracts;
|
|
quest.conditions.AvailableForFinish[0].id = this.objectId.generate();
|
|
quest.location = this.getQuestLocationByMapId(locationKey);
|
|
|
|
if (requiresSpecificExtract)
|
|
{
|
|
// Filter by whitelist, it's also possible that the field "PassageRequirement" does not exist (e.g. Shoreline)
|
|
let mapExits = this.getLocationExitsForSide(locationKey, repeatableConfig.side);
|
|
|
|
// Exclude scav coop exits when choosing pmc exit
|
|
if (repeatableConfig.side === "Pmc")
|
|
{
|
|
mapExits = mapExits.filter((exit) => exit.PassageRequirement !== "ScavCooperation");
|
|
}
|
|
|
|
// Only get exits that have a greater than 0% chance to spawn
|
|
const exitPool = mapExits.filter((exit) => exit.Chance > 0);
|
|
|
|
// Exclude exits with a requirement to leave (e.g. car extracts)
|
|
const possibleExits = exitPool.filter((
|
|
exit,
|
|
) => (!("PassageRequirement" in exit)
|
|
|| repeatableConfig.questConfig.Exploration.specificExits.passageRequirementWhitelist.includes(
|
|
exit.PassageRequirement,
|
|
))
|
|
);
|
|
|
|
if (possibleExits.length === 0)
|
|
{
|
|
this.logger.error(
|
|
`Unable to choose specific exit on map: ${locationKey}, Possible exit pool was empty`,
|
|
);
|
|
}
|
|
else
|
|
{
|
|
// Choose one of the exits we filtered above
|
|
const chosenExit = this.randomUtil.drawRandomFromList(possibleExits, 1)[0];
|
|
|
|
// Create a quest condition to leave raid via chosen exit
|
|
const exitCondition = this.generateExplorationExitCondition(chosenExit);
|
|
quest.conditions.AvailableForFinish[0].counter.conditions.push(exitCondition);
|
|
}
|
|
}
|
|
|
|
// Difficulty for exploration goes from 1 extract to maxExtracts
|
|
// Difficulty for reward goes from 0.2...1 -> map
|
|
const difficulty = this.mathUtil.mapToRange(numExtracts, 1, explorationConfig.maxExtracts, 0.2, 1);
|
|
quest.rewards = this.generateReward(pmcLevel, difficulty, traderId, repeatableConfig, explorationConfig);
|
|
|
|
return quest;
|
|
}
|
|
|
|
/**
|
|
* Filter a maps exits to just those for the desired side
|
|
* @param locationKey Map id (e.g. factory4_day)
|
|
* @param playerSide Scav/Bear
|
|
* @returns Array of Exit objects
|
|
*/
|
|
protected getLocationExitsForSide(locationKey: string, playerSide: string): Exit[]
|
|
{
|
|
const mapBase = this.databaseServer.getTables().locations[locationKey.toLowerCase()].base as ILocationBase;
|
|
|
|
const infilPointsOfSameSide = new Set<string>();
|
|
for (const spawnPoint of mapBase.SpawnPointParams)
|
|
{
|
|
// Same side, add infil to list
|
|
if (spawnPoint.Sides.includes(playerSide) || spawnPoint.Sides.includes("All"))
|
|
{
|
|
// Has specific start location
|
|
if (spawnPoint.Infiltration.length > 0)
|
|
{
|
|
infilPointsOfSameSide.add(spawnPoint.Infiltration);
|
|
}
|
|
}
|
|
}
|
|
|
|
// use list of allowed infils to figure out side of exits
|
|
const infilPointsArray = Array.from(infilPointsOfSameSide);
|
|
|
|
return mapBase.exits.filter((exit) =>
|
|
exit.EntryPoints.split(",").some((entryPoint) => infilPointsArray.includes(entryPoint))
|
|
);
|
|
}
|
|
|
|
protected generatePickupQuest(
|
|
pmcLevel: number,
|
|
traderId: string,
|
|
questTypePool: IQuestTypePool,
|
|
repeatableConfig: IRepeatableQuestConfig,
|
|
): IRepeatableQuest
|
|
{
|
|
const pickupConfig = repeatableConfig.questConfig.Pickup;
|
|
|
|
const quest = this.generateRepeatableTemplate("Pickup", traderId, repeatableConfig.side);
|
|
|
|
const itemTypeToFetchWithCount = this.randomUtil.getArrayValue(pickupConfig.ItemTypeToFetchWithMaxCount);
|
|
const itemCountToFetch = this.randomUtil.randInt(
|
|
itemTypeToFetchWithCount.minPickupCount,
|
|
itemTypeToFetchWithCount.maxPickupCount + 1,
|
|
);
|
|
// Choose location - doesnt seem to work for anything other than 'any'
|
|
// const locationKey: string = this.randomUtil.drawRandomFromDict(questTypePool.pool.Pickup.locations)[0];
|
|
// const locationTarget = questTypePool.pool.Pickup.locations[locationKey];
|
|
|
|
const findCondition = quest.conditions.AvailableForFinish.find((x) => x.conditionType === "FindItem");
|
|
findCondition.target = [itemTypeToFetchWithCount.itemType];
|
|
findCondition.value = itemCountToFetch;
|
|
|
|
const counterCreatorCondition = quest.conditions.AvailableForFinish.find((x) =>
|
|
x.conditionType === "CounterCreator"
|
|
);
|
|
// const locationCondition = counterCreatorCondition._props.counter.conditions.find(x => x._parent === "Location");
|
|
// (locationCondition._props as ILocationConditionProps).target = [...locationTarget];
|
|
|
|
const equipmentCondition = counterCreatorCondition.counter.conditions.find((x) =>
|
|
x.conditionType === "Equipment"
|
|
);
|
|
equipmentCondition.equipmentInclusive = [[itemTypeToFetchWithCount.itemType]];
|
|
|
|
// Add rewards
|
|
quest.rewards = this.generateReward(pmcLevel, 1, traderId, repeatableConfig, pickupConfig);
|
|
|
|
return quest;
|
|
}
|
|
|
|
/**
|
|
* Convert a location into an quest code can read (e.g. factory4_day into 55f2d3fd4bdc2d5f408b4567)
|
|
* @param locationKey e.g factory4_day
|
|
* @returns guid
|
|
*/
|
|
protected getQuestLocationByMapId(locationKey: string): string
|
|
{
|
|
return this.questConfig.locationIdMap[locationKey];
|
|
}
|
|
|
|
/**
|
|
* Exploration repeatable quests can specify a required extraction point.
|
|
* This method creates the according object which will be appended to the conditions array
|
|
*
|
|
* @param {string} exit The exit name to generate the condition for
|
|
* @returns {object} Exit condition
|
|
*/
|
|
protected generateExplorationExitCondition(exit: Exit): IQuestConditionCounterCondition
|
|
{
|
|
return { conditionType: "ExitName", exitName: exit.Name, id: this.objectId.generate(), dynamicLocale: true };
|
|
}
|
|
|
|
/**
|
|
* Generate the reward for a mission. A reward can consist of
|
|
* - Experience
|
|
* - Money
|
|
* - Items
|
|
* - Trader Reputation
|
|
*
|
|
* The reward is dependent on the player level as given by the wiki. The exact mapping of pmcLevel to
|
|
* experience / money / items / trader reputation can be defined in QuestConfig.js
|
|
*
|
|
* There's also a random variation of the reward the spread of which can be also defined in the config.
|
|
*
|
|
* Additionally, a scaling factor w.r.t. quest difficulty going from 0.2...1 can be used
|
|
*
|
|
* @param {integer} pmcLevel player's level
|
|
* @param {number} difficulty a reward scaling factor from 0.2 to 1
|
|
* @param {string} traderId the trader for reputation gain (and possible in the future filtering of reward item type based on trader)
|
|
* @param {object} repeatableConfig The configuration for the repeatable kind (daily, weekly) as configured in QuestConfig for the requested quest
|
|
* @returns {object} object of "Reward"-type that can be given for a repeatable mission
|
|
*/
|
|
protected generateReward(
|
|
pmcLevel: number,
|
|
difficulty: number,
|
|
traderId: string,
|
|
repeatableConfig: IRepeatableQuestConfig,
|
|
questConfig: IBaseQuestConfig,
|
|
): IQuestRewards
|
|
{
|
|
// difficulty could go from 0.2 ... -> for lowest difficulty receive 0.2*nominal reward
|
|
const levelsConfig = repeatableConfig.rewardScaling.levels;
|
|
const roublesConfig = repeatableConfig.rewardScaling.roubles;
|
|
const xpConfig = repeatableConfig.rewardScaling.experience;
|
|
const itemsConfig = repeatableConfig.rewardScaling.items;
|
|
const rewardSpreadConfig = repeatableConfig.rewardScaling.rewardSpread;
|
|
const skillRewardChanceConfig = repeatableConfig.rewardScaling.skillRewardChance;
|
|
const skillPointRewardConfig = repeatableConfig.rewardScaling.skillPointReward;
|
|
const reputationConfig = repeatableConfig.rewardScaling.reputation;
|
|
|
|
const effectiveDifficulty = Number.isNaN(difficulty) ? 1 : difficulty;
|
|
if (Number.isNaN(difficulty))
|
|
{
|
|
this.logger.warning(this.localisationService.getText("repeatable-difficulty_was_nan"));
|
|
}
|
|
|
|
// rewards are generated based on pmcLevel, difficulty and a random spread
|
|
const rewardXP = Math.floor(
|
|
effectiveDifficulty * this.mathUtil.interp1(pmcLevel, levelsConfig, xpConfig)
|
|
* this.randomUtil.getFloat(1 - rewardSpreadConfig, 1 + rewardSpreadConfig),
|
|
);
|
|
const rewardRoubles = Math.floor(
|
|
effectiveDifficulty * this.mathUtil.interp1(pmcLevel, levelsConfig, roublesConfig)
|
|
* this.randomUtil.getFloat(1 - rewardSpreadConfig, 1 + rewardSpreadConfig),
|
|
);
|
|
const rewardNumItems = this.randomUtil.randInt(
|
|
1,
|
|
Math.round(this.mathUtil.interp1(pmcLevel, levelsConfig, itemsConfig)) + 1,
|
|
);
|
|
const rewardReputation =
|
|
Math.round(
|
|
100 * effectiveDifficulty * this.mathUtil.interp1(pmcLevel, levelsConfig, reputationConfig)
|
|
* this.randomUtil.getFloat(1 - rewardSpreadConfig, 1 + rewardSpreadConfig),
|
|
) / 100;
|
|
const skillRewardChance = this.mathUtil.interp1(pmcLevel, levelsConfig, skillRewardChanceConfig);
|
|
const skillPointReward = this.mathUtil.interp1(pmcLevel, levelsConfig, skillPointRewardConfig);
|
|
|
|
// Possible improvement -> draw trader-specific items e.g. with this.itemHelper.isOfBaseclass(val._id, ItemHelper.BASECLASS.FoodDrink)
|
|
let roublesBudget = rewardRoubles;
|
|
let rewardItemPool = this.chooseRewardItemsWithinBudget(repeatableConfig, roublesBudget, traderId);
|
|
|
|
const rewards: IQuestRewards = { Started: [], Success: [], Fail: [] };
|
|
|
|
let rewardIndex = 0;
|
|
// Add xp reward
|
|
if (rewardXP > 0)
|
|
{
|
|
rewards.Success.push({ value: rewardXP, type: QuestRewardType.EXPERIENCE, index: rewardIndex });
|
|
rewardIndex++;
|
|
}
|
|
|
|
// Add money reward
|
|
this.addMoneyReward(traderId, rewards, rewardRoubles, rewardIndex);
|
|
rewardIndex++;
|
|
|
|
const traderWhitelistDetails = repeatableConfig.traderWhitelist.find((x) => x.traderId === traderId);
|
|
if (
|
|
traderWhitelistDetails.rewardCanBeWeapon
|
|
&& this.randomUtil.getChance100(traderWhitelistDetails.weaponRewardChancePercent)
|
|
)
|
|
{
|
|
// Add a random default preset weapon as reward
|
|
const defaultPresetPool = new ExhaustableArray(
|
|
Object.values(this.presetHelper.getDefaultWeaponPresets()),
|
|
this.randomUtil,
|
|
this.jsonUtil,
|
|
);
|
|
let chosenPreset: IPreset;
|
|
while (defaultPresetPool.hasValues())
|
|
{
|
|
const randomPreset = defaultPresetPool.getRandomValue();
|
|
const tpls = randomPreset._items.map((item) => item._tpl);
|
|
const presetPrice = this.itemHelper.getItemAndChildrenPrice(tpls);
|
|
if (presetPrice <= roublesBudget)
|
|
{
|
|
chosenPreset = this.jsonUtil.clone(randomPreset);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (chosenPreset)
|
|
{
|
|
// use _encyclopedia as its always the base items _tpl, items[0] isn't guaranteed to be base item
|
|
rewards.Success.push(
|
|
this.generateRewardItem(chosenPreset._encyclopedia, 1, rewardIndex, chosenPreset._items),
|
|
);
|
|
rewardIndex++;
|
|
}
|
|
}
|
|
|
|
if (rewardItemPool.length > 0)
|
|
{
|
|
for (let i = 0; i < rewardNumItems; i++)
|
|
{
|
|
let rewardItemStackCount = 1;
|
|
const itemSelected = rewardItemPool[this.randomUtil.randInt(rewardItemPool.length)];
|
|
|
|
if (this.itemHelper.isOfBaseclass(itemSelected._id, BaseClasses.AMMO))
|
|
{
|
|
// Don't reward ammo that stacks to less than what's defined in config
|
|
if (itemSelected._props.StackMaxSize < repeatableConfig.rewardAmmoStackMinSize)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
// Choose smallest value between budget fitting size and stack max
|
|
rewardItemStackCount = this.calculateAmmoStackSizeThatFitsBudget(
|
|
itemSelected,
|
|
roublesBudget,
|
|
rewardNumItems,
|
|
);
|
|
}
|
|
|
|
// 25% chance to double, triple quadruple reward stack (Only occurs when item is stackable and not weapon or ammo)
|
|
if (this.canIncreaseRewardItemStackSize(itemSelected, 70000))
|
|
{
|
|
rewardItemStackCount = this.getRandomisedRewardItemStackSizeByPrice(itemSelected);
|
|
}
|
|
|
|
rewards.Success.push(this.generateRewardItem(itemSelected._id, rewardItemStackCount, rewardIndex));
|
|
rewardIndex++;
|
|
|
|
const itemCost = this.itemHelper.getStaticItemPrice(itemSelected._id);
|
|
roublesBudget -= rewardItemStackCount * itemCost;
|
|
|
|
// If we still have budget narrow down possible items
|
|
if (roublesBudget > 0)
|
|
{
|
|
// Filter possible reward items to only items with a price below the remaining budget
|
|
rewardItemPool = rewardItemPool.filter((x) =>
|
|
this.itemHelper.getStaticItemPrice(x._id) < roublesBudget
|
|
);
|
|
if (rewardItemPool.length === 0)
|
|
{
|
|
break; // No reward items left, exit
|
|
}
|
|
}
|
|
else
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Add rep reward to rewards array
|
|
if (rewardReputation > 0)
|
|
{
|
|
const reward: IQuestReward = {
|
|
target: traderId,
|
|
value: rewardReputation,
|
|
type: QuestRewardType.TRADER_STANDING,
|
|
index: rewardIndex,
|
|
};
|
|
rewards.Success.push(reward);
|
|
rewardIndex++;
|
|
}
|
|
|
|
// Chance of adding skill reward
|
|
if (this.randomUtil.getChance100(skillRewardChance * 100))
|
|
{
|
|
const reward: IQuestReward = {
|
|
target: this.randomUtil.getArrayValue(questConfig.possibleSkillRewards),
|
|
value: skillPointReward,
|
|
type: QuestRewardType.SKILL,
|
|
index: rewardIndex,
|
|
};
|
|
rewards.Success.push(reward);
|
|
}
|
|
|
|
return rewards;
|
|
}
|
|
|
|
protected addMoneyReward(traderId: string, rewards: IQuestRewards, rewardRoubles: number, rewardIndex: number): void
|
|
{
|
|
// PK and Fence use euros
|
|
if (traderId === Traders.PEACEKEEPER || traderId === Traders.FENCE)
|
|
{
|
|
rewards.Success.push(
|
|
this.generateRewardItem(
|
|
Money.EUROS,
|
|
this.handbookHelper.fromRUB(rewardRoubles, Money.EUROS),
|
|
rewardIndex,
|
|
),
|
|
);
|
|
}
|
|
else
|
|
{
|
|
// Everyone else uses roubles
|
|
rewards.Success.push(this.generateRewardItem(Money.ROUBLES, rewardRoubles, rewardIndex));
|
|
}
|
|
}
|
|
|
|
protected calculateAmmoStackSizeThatFitsBudget(
|
|
itemSelected: ITemplateItem,
|
|
roublesBudget: number,
|
|
rewardNumItems: number,
|
|
): number
|
|
{
|
|
// The budget for this ammo stack
|
|
const stackRoubleBudget = roublesBudget / rewardNumItems;
|
|
|
|
const singleCartridgePrice = this.handbookHelper.getTemplatePrice(itemSelected._id);
|
|
|
|
// Get a stack size of ammo that fits rouble budget
|
|
const stackSizeThatFitsBudget = Math.round(stackRoubleBudget / singleCartridgePrice);
|
|
|
|
// Get itemDbs max stack size for ammo - don't go above 100 (some mods mess around with stack sizes)
|
|
const stackMaxCount = Math.min(itemSelected._props.StackMaxSize, 100);
|
|
|
|
return Math.min(stackSizeThatFitsBudget, stackMaxCount);
|
|
}
|
|
|
|
/**
|
|
* Should reward item have stack size increased (25% chance)
|
|
* @param item Item to possibly increase stack size of
|
|
* @param maxRoublePriceToStack Maximum rouble price an item can be to still be chosen for stacking
|
|
* @returns True if it should
|
|
*/
|
|
protected canIncreaseRewardItemStackSize(item: ITemplateItem, maxRoublePriceToStack: number): boolean
|
|
{
|
|
return this.itemHelper.getStaticItemPrice(item._id) < maxRoublePriceToStack
|
|
&& !this.itemHelper.isOfBaseclasses(item._id, [BaseClasses.WEAPON, BaseClasses.AMMO])
|
|
&& !this.itemHelper.itemRequiresSoftInserts(item._id)
|
|
&& this.randomUtil.getChance100(25);
|
|
}
|
|
|
|
/**
|
|
* Get a randomised number a reward items stack size should be based on its handbook price
|
|
* @param item Reward item to get stack size for
|
|
* @returns Stack size value
|
|
*/
|
|
protected getRandomisedRewardItemStackSizeByPrice(item: ITemplateItem): number
|
|
{
|
|
const rewardItemPrice = this.itemHelper.getStaticItemPrice(item._id);
|
|
if (rewardItemPrice < 3000)
|
|
{
|
|
return this.randomUtil.getArrayValue([2, 3, 4]);
|
|
}
|
|
|
|
if (rewardItemPrice < 10000)
|
|
{
|
|
return this.randomUtil.getArrayValue([2, 3]);
|
|
}
|
|
|
|
return 2;
|
|
}
|
|
|
|
/**
|
|
* Select a number of items that have a colelctive value of the passed in parameter
|
|
* @param repeatableConfig Config
|
|
* @param roublesBudget Total value of items to return
|
|
* @returns Array of reward items that fit budget
|
|
*/
|
|
protected chooseRewardItemsWithinBudget(
|
|
repeatableConfig: IRepeatableQuestConfig,
|
|
roublesBudget: number,
|
|
traderId: string,
|
|
): ITemplateItem[]
|
|
{
|
|
// First filter for type and baseclass to avoid lookup in handbook for non-available items
|
|
const rewardableItemPool = this.getRewardableItems(repeatableConfig, traderId);
|
|
const minPrice = Math.min(25000, 0.5 * roublesBudget);
|
|
|
|
let rewardableItemPoolWithinBudget = rewardableItemPool.filter((item) =>
|
|
{
|
|
// Get default preset if it exists
|
|
const defaultPreset = this.presetHelper.getDefaultPreset(item[0]);
|
|
|
|
// Bundle up tpls we want price for
|
|
const tpls = defaultPreset ? defaultPreset._items.map((item) => item._tpl) : [item[0]];
|
|
|
|
// Get price of tpls
|
|
const itemPrice = this.itemHelper.getItemAndChildrenPrice(tpls);
|
|
|
|
return itemPrice < roublesBudget && itemPrice > minPrice;
|
|
}).map((x) => x[1]);
|
|
|
|
if (rewardableItemPoolWithinBudget.length === 0)
|
|
{
|
|
this.logger.warning(
|
|
this.localisationService.getText("repeatable-no_reward_item_found_in_price_range", {
|
|
minPrice: minPrice,
|
|
roublesBudget: roublesBudget,
|
|
}),
|
|
);
|
|
// In case we don't find any items in the price range
|
|
rewardableItemPoolWithinBudget = rewardableItemPool.filter((x) =>
|
|
this.itemHelper.getItemPrice(x[0]) < roublesBudget
|
|
).map((x) => x[1]);
|
|
}
|
|
|
|
return rewardableItemPoolWithinBudget;
|
|
}
|
|
|
|
/**
|
|
* Helper to create a reward item structured as required by the client
|
|
*
|
|
* @param {string} tpl ItemId of the rewarded item
|
|
* @param {integer} value Amount of items to give
|
|
* @param {integer} index All rewards will be appended to a list, for unknown reasons the client wants the index
|
|
* @returns {object} Object of "Reward"-item-type
|
|
*/
|
|
protected generateRewardItem(tpl: string, value: number, index: number, preset: Item[] = null): IQuestReward
|
|
{
|
|
const id = this.objectId.generate();
|
|
const rewardItem: IQuestReward = { target: id, value: value, type: QuestRewardType.ITEM, index: index };
|
|
|
|
if (preset)
|
|
{
|
|
const rootItem = preset.find((x) => x._tpl === tpl);
|
|
rewardItem.items = this.itemHelper.reparentItemAndChildren(rootItem, preset);
|
|
rewardItem.target = rootItem._id; // Target property and root items id must match
|
|
}
|
|
else
|
|
{
|
|
const rootItem = { _id: id, _tpl: tpl, upd: { StackObjectsCount: value, SpawnedInSession: true } };
|
|
rewardItem.items = [rootItem];
|
|
}
|
|
return rewardItem;
|
|
}
|
|
|
|
/**
|
|
* Picks rewardable items from items.json. This means they need to fit into the inventory and they shouldn't be keys (debatable)
|
|
* @param repeatableQuestConfig Config file
|
|
* @returns List of rewardable items [[_tpl, itemTemplate],...]
|
|
*/
|
|
protected getRewardableItems(
|
|
repeatableQuestConfig: IRepeatableQuestConfig,
|
|
traderId: string,
|
|
): [string, ITemplateItem][]
|
|
{
|
|
// Get an array of seasonal items that should not be shown right now as seasonal event is not active
|
|
const seasonalItems = this.seasonalEventService.getInactiveSeasonalEventItems();
|
|
|
|
// check for specific baseclasses which don't make sense as reward item
|
|
// also check if the price is greater than 0; there are some items whose price can not be found
|
|
// those are not in the game yet (e.g. AGS grenade launcher)
|
|
return Object.entries(this.databaseServer.getTables().templates.items).filter(
|
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
([tpl, itemTemplate]) =>
|
|
{
|
|
// Base "Item" item has no parent, ignore it
|
|
if (itemTemplate._parent === "")
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (seasonalItems.includes(tpl))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
const traderWhitelist = repeatableQuestConfig.traderWhitelist.find((trader) =>
|
|
trader.traderId === traderId
|
|
);
|
|
return this.isValidRewardItem(tpl, repeatableQuestConfig, traderWhitelist?.rewardBaseWhitelist);
|
|
},
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Checks if an id is a valid item. Valid meaning that it's an item that may be a reward
|
|
* or content of bot loot. Items that are tested as valid may be in a player backpack or stash.
|
|
* @param {string} tpl template id of item to check
|
|
* @returns True if item is valid reward
|
|
*/
|
|
protected isValidRewardItem(
|
|
tpl: string,
|
|
repeatableQuestConfig: IRepeatableQuestConfig,
|
|
itemBaseWhitelist: string[],
|
|
): boolean
|
|
{
|
|
if (!this.itemHelper.isValidItem(tpl))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
// Check global blacklist
|
|
if (this.itemFilterService.isItemBlacklisted(tpl))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
// Item is on repeatable or global blacklist
|
|
if (repeatableQuestConfig.rewardBlacklist.includes(tpl) || this.itemFilterService.isItemBlacklisted(tpl))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
// Item has blacklisted base type
|
|
if (this.itemHelper.isOfBaseclasses(tpl, [...repeatableQuestConfig.rewardBaseTypeBlacklist]))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
// Skip boss items
|
|
if (this.itemFilterService.isBossItem(tpl))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
// Trader has specific item base types they can give as rewards to player
|
|
if (itemBaseWhitelist !== undefined)
|
|
{
|
|
if (!this.itemHelper.isOfBaseclasses(tpl, [...itemBaseWhitelist]))
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Generates the base object of quest type format given as templates in assets/database/templates/repeatableQuests.json
|
|
* The templates include Elimination, Completion and Extraction quest types
|
|
*
|
|
* @param {string} type Quest type: "Elimination", "Completion" or "Extraction"
|
|
* @param {string} traderId Trader from which the quest will be provided
|
|
* @param {string} side Scav daily or pmc daily/weekly quest
|
|
* @returns {object} Object which contains the base elements for repeatable quests of the requests type
|
|
* (needs to be filled with reward and conditions by called to make a valid quest)
|
|
*/
|
|
// @Incomplete: define Type for "type".
|
|
protected generateRepeatableTemplate(type: string, traderId: string, side: string): IRepeatableQuest
|
|
{
|
|
const questClone = this.jsonUtil.clone<IRepeatableQuest>(
|
|
this.databaseServer.getTables().templates.repeatableQuests.templates[type],
|
|
);
|
|
questClone._id = this.objectId.generate();
|
|
questClone.traderId = traderId;
|
|
|
|
/* in locale, these id correspond to the text of quests
|
|
template ids -pmc : Elimination = 616052ea3054fc0e2c24ce6e / Completion = 61604635c725987e815b1a46 / Exploration = 616041eb031af660100c9967
|
|
template ids -scav : Elimination = 62825ef60e88d037dc1eb428 / Completion = 628f588ebb558574b2260fe5 / Exploration = 62825ef60e88d037dc1eb42c
|
|
*/
|
|
|
|
// Get template id from config based on side and type of quest
|
|
questClone.templateId = this.questConfig.questTemplateIds[side.toLowerCase()][type.toLowerCase()];
|
|
|
|
questClone.name = questClone.name.replace("{traderId}", traderId).replace(
|
|
"{templateId}",
|
|
questClone.templateId,
|
|
);
|
|
questClone.note = questClone.note.replace("{traderId}", traderId).replace(
|
|
"{templateId}",
|
|
questClone.templateId,
|
|
);
|
|
questClone.description = questClone.description.replace("{traderId}", traderId).replace(
|
|
"{templateId}",
|
|
questClone.templateId,
|
|
);
|
|
questClone.successMessageText = questClone.successMessageText.replace("{traderId}", traderId).replace(
|
|
"{templateId}",
|
|
questClone.templateId,
|
|
);
|
|
questClone.failMessageText = questClone.failMessageText.replace("{traderId}", traderId).replace(
|
|
"{templateId}",
|
|
questClone.templateId,
|
|
);
|
|
questClone.startedMessageText = questClone.startedMessageText.replace("{traderId}", traderId).replace(
|
|
"{templateId}",
|
|
questClone.templateId,
|
|
);
|
|
questClone.changeQuestMessageText = questClone.changeQuestMessageText.replace("{traderId}", traderId).replace(
|
|
"{templateId}",
|
|
questClone.templateId,
|
|
);
|
|
questClone.acceptPlayerMessage = questClone.acceptPlayerMessage.replace("{traderId}", traderId).replace(
|
|
"{templateId}",
|
|
questClone.templateId,
|
|
);
|
|
questClone.declinePlayerMessage = questClone.declinePlayerMessage.replace("{traderId}", traderId).replace(
|
|
"{templateId}",
|
|
questClone.templateId,
|
|
);
|
|
questClone.completePlayerMessage = questClone.completePlayerMessage.replace("{traderId}", traderId).replace(
|
|
"{templateId}",
|
|
questClone.templateId,
|
|
);
|
|
|
|
return questClone;
|
|
}
|
|
}
|