2023-03-03 16:23:46 +01:00
import { inject , injectable } from "tsyringe" ;
2024-05-21 19:59:04 +02:00
import { DialogueHelper } from "@spt/helpers/DialogueHelper" ;
import { ItemHelper } from "@spt/helpers/ItemHelper" ;
import { PaymentHelper } from "@spt/helpers/PaymentHelper" ;
import { PresetHelper } from "@spt/helpers/PresetHelper" ;
import { ProfileHelper } from "@spt/helpers/ProfileHelper" ;
import { QuestConditionHelper } from "@spt/helpers/QuestConditionHelper" ;
import { RagfairServerHelper } from "@spt/helpers/RagfairServerHelper" ;
import { TraderHelper } from "@spt/helpers/TraderHelper" ;
import { IPmcData } from "@spt/models/eft/common/IPmcData" ;
import { Common , IQuestStatus } from "@spt/models/eft/common/tables/IBotBase" ;
import { Item } from "@spt/models/eft/common/tables/IItem" ;
import { IQuest , IQuestCondition , IQuestReward } from "@spt/models/eft/common/tables/IQuest" ;
import { IItemEventRouterResponse } from "@spt/models/eft/itemEvent/IItemEventRouterResponse" ;
import { IAcceptQuestRequestData } from "@spt/models/eft/quests/IAcceptQuestRequestData" ;
import { IFailQuestRequestData } from "@spt/models/eft/quests/IFailQuestRequestData" ;
import { ConfigTypes } from "@spt/models/enums/ConfigTypes" ;
import { MessageType } from "@spt/models/enums/MessageType" ;
import { QuestRewardType } from "@spt/models/enums/QuestRewardType" ;
import { QuestStatus } from "@spt/models/enums/QuestStatus" ;
import { SkillTypes } from "@spt/models/enums/SkillTypes" ;
import { IQuestConfig } from "@spt/models/spt/config/IQuestConfig" ;
import { ILogger } from "@spt/models/spt/utils/ILogger" ;
import { EventOutputHolder } from "@spt/routers/EventOutputHolder" ;
import { ConfigServer } from "@spt/servers/ConfigServer" ;
2024-05-29 16:15:45 +02:00
import { DatabaseService } from "@spt/services/DatabaseService" ;
2024-05-21 19:59:04 +02:00
import { LocaleService } from "@spt/services/LocaleService" ;
import { LocalisationService } from "@spt/services/LocalisationService" ;
import { MailSendService } from "@spt/services/MailSendService" ;
import { ICloner } from "@spt/utils/cloners/ICloner" ;
import { HashUtil } from "@spt/utils/HashUtil" ;
import { TimeUtil } from "@spt/utils/TimeUtil" ;
2023-03-03 16:23:46 +01:00
@injectable ( )
export class QuestHelper
{
protected questConfig : IQuestConfig ;
constructor (
2024-05-28 16:04:20 +02:00
@inject ( "PrimaryLogger" ) protected logger : ILogger ,
2023-03-03 16:23:46 +01:00
@inject ( "TimeUtil" ) protected timeUtil : TimeUtil ,
@inject ( "HashUtil" ) protected hashUtil : HashUtil ,
@inject ( "ItemHelper" ) protected itemHelper : ItemHelper ,
2024-05-29 16:15:45 +02:00
@inject ( "DatabaseService" ) protected databaseService : DatabaseService ,
2023-10-10 13:03:20 +02:00
@inject ( "QuestConditionHelper" ) protected questConditionHelper : QuestConditionHelper ,
2023-03-03 16:23:46 +01:00
@inject ( "EventOutputHolder" ) protected eventOutputHolder : EventOutputHolder ,
@inject ( "LocaleService" ) protected localeService : LocaleService ,
@inject ( "RagfairServerHelper" ) protected ragfairServerHelper : RagfairServerHelper ,
@inject ( "DialogueHelper" ) protected dialogueHelper : DialogueHelper ,
@inject ( "ProfileHelper" ) protected profileHelper : ProfileHelper ,
@inject ( "PaymentHelper" ) protected paymentHelper : PaymentHelper ,
@inject ( "LocalisationService" ) protected localisationService : LocalisationService ,
@inject ( "TraderHelper" ) protected traderHelper : TraderHelper ,
2024-01-10 11:04:09 +01:00
@inject ( "PresetHelper" ) protected presetHelper : PresetHelper ,
2023-07-22 14:02:42 +02:00
@inject ( "MailSendService" ) protected mailSendService : MailSendService ,
2023-11-16 22:42:06 +01:00
@inject ( "ConfigServer" ) protected configServer : ConfigServer ,
2024-05-28 16:04:20 +02:00
@inject ( "PrimaryCloner" ) protected cloner : ICloner ,
2023-03-03 16:23:46 +01:00
)
{
this . questConfig = this . configServer . getConfig ( ConfigTypes . QUEST ) ;
}
/ * *
2023-11-16 22:42:06 +01:00
* Get status of a quest in player profile by its id
* @param pmcData Profile to search
* @param questId Quest id to look up
* @returns QuestStatus enum
* /
2023-03-21 15:19:49 +01:00
public getQuestStatus ( pmcData : IPmcData , questId : string ) : QuestStatus
2023-03-03 16:23:46 +01:00
{
2024-05-17 21:32:41 +02:00
const quest = pmcData . Quests ? . find ( ( q ) = > q . qid === questId ) ;
2023-03-03 16:23:46 +01:00
2023-11-16 22:42:06 +01:00
return quest ? quest.status : QuestStatus.Locked ;
2023-03-03 16:23:46 +01:00
}
/ * *
* returns true is the level condition is satisfied
* @param playerLevel Players level
* @param condition Quest condition
* @returns true if player level is greater than or equal to quest
* /
2024-01-05 20:52:21 +01:00
public doesPlayerLevelFulfilCondition ( playerLevel : number , condition : IQuestCondition ) : boolean
2023-03-03 16:23:46 +01:00
{
2023-12-27 18:15:38 +01:00
if ( condition . conditionType === "Level" )
2023-03-03 16:23:46 +01:00
{
2023-12-27 18:15:38 +01:00
switch ( condition . compareMethod )
2023-03-03 16:23:46 +01:00
{
case ">=" :
2023-12-27 18:15:38 +01:00
return playerLevel >= < number > condition . value ;
2023-05-22 17:00:09 +02:00
case ">" :
2023-12-27 18:15:38 +01:00
return playerLevel > < number > condition . value ;
2023-05-22 17:00:09 +02:00
case "<" :
2023-12-27 18:15:38 +01:00
return playerLevel < < number > condition . value ;
2023-05-22 17:00:09 +02:00
case "<=" :
2023-12-27 18:15:38 +01:00
return playerLevel <= < number > condition . value ;
2023-05-22 17:00:09 +02:00
case "=" :
2023-12-27 18:15:38 +01:00
return playerLevel === < number > condition . value ;
2023-03-03 16:23:46 +01:00
default :
2023-11-16 22:42:06 +01:00
this . logger . error (
this . localisationService . getText (
"quest-unable_to_find_compare_condition" ,
2023-12-27 18:15:38 +01:00
condition . compareMethod ,
2023-11-16 22:42:06 +01:00
) ,
) ;
2023-03-03 16:23:46 +01:00
return false ;
}
}
}
/ * *
* Get the quests found in both arrays ( inner join )
* @param before Array of quests # 1
* @param after Array of quests # 2
* @returns Reduction of cartesian product between two quest arrays
* /
public getDeltaQuests ( before : IQuest [ ] , after : IQuest [ ] ) : IQuest [ ]
{
const knownQuestsIds = [ ] ;
for ( const q of before )
{
knownQuestsIds . push ( q . _id ) ;
}
if ( knownQuestsIds . length )
{
return after . filter ( ( q ) = >
{
return knownQuestsIds . indexOf ( q . _id ) === - 1 ;
} ) ;
}
return after ;
}
2023-11-02 09:56:02 +01:00
/ * *
* Adjust skill experience for low skill levels , mimicing the official client
* @param profileSkill the skill experience is being added to
* @param progressAmount the amount of experience being added to the skill
* @returns the adjusted skill progress gain
* /
public adjustSkillExpForLowLevels ( profileSkill : Common , progressAmount : number ) : number
{
let currentLevel = Math . floor ( profileSkill . Progress / 100 ) ;
// Only run this if the current level is under 9
if ( currentLevel >= 9 )
{
return progressAmount ;
}
// This calculates how much progress we have in the skill's starting level
2024-05-17 21:32:41 +02:00
let startingLevelProgress = ( profileSkill . Progress % 100 ) * ( ( currentLevel + 1 ) / 10 ) ;
2023-11-02 09:56:02 +01:00
// The code below assumes a 1/10th progress skill amount
let remainingProgress = progressAmount / 10 ;
// We have to do this loop to handle edge cases where the provided XP bumps your level up
// See "CalculateExpOnFirstLevels" in client for original logic
let adjustedSkillProgress = 0 ;
while ( remainingProgress > 0 && currentLevel < 9 )
{
// Calculate how much progress to add, limiting it to the current level max progress
2024-05-08 05:57:08 +02:00
const currentLevelRemainingProgress = ( currentLevel + 1 ) * 10 - startingLevelProgress ;
2023-11-02 09:56:02 +01:00
this . logger . debug ( ` currentLevelRemainingProgress: ${ currentLevelRemainingProgress } ` ) ;
const progressToAdd = Math . min ( remainingProgress , currentLevelRemainingProgress ) ;
2024-05-17 21:32:41 +02:00
const adjustedProgressToAdd = ( 10 / ( currentLevel + 1 ) ) * progressToAdd ;
2023-11-02 09:56:02 +01:00
this . logger . debug ( ` Progress To Add: ${ progressToAdd } Adjusted for level: ${ adjustedProgressToAdd } ` ) ;
// Add the progress amount adjusted by level
adjustedSkillProgress += adjustedProgressToAdd ;
remainingProgress -= progressToAdd ;
startingLevelProgress = 0 ;
currentLevel ++ ;
}
// If there's any remaining progress, add it. This handles if you go from level 8 -> 9
if ( remainingProgress > 0 )
{
adjustedSkillProgress += remainingProgress ;
}
return adjustedSkillProgress ;
}
2023-03-03 16:23:46 +01:00
/ * *
* Get quest name by quest id
* @param questId id to get
2023-11-16 22:42:06 +01:00
* @returns
2023-03-03 16:23:46 +01:00
* /
public getQuestNameFromLocale ( questId : string ) : string
{
const questNameKey = ` ${ questId } name ` ;
return this . localeService . getLocaleDb ( ) [ questNameKey ] ;
}
/ * *
* Check if trader has sufficient loyalty to fulfill quest requirement
* @param questProperties Quest props
* @param profile Player profile
* @returns true if loyalty is high enough to fulfill quest requirement
* /
2024-01-05 20:52:21 +01:00
public traderLoyaltyLevelRequirementCheck ( questProperties : IQuestCondition , profile : IPmcData ) : boolean
2023-10-10 13:03:20 +02:00
{
const requiredLoyaltyLevel = Number ( questProperties . value ) ;
const trader = profile . TradersInfo [ < string > questProperties . target ] ;
if ( ! trader )
{
2024-05-21 13:40:16 +02:00
this . logger . error ( this . localisationService . getText ( "quest-unable_to_find_trader_in_profile" , questProperties . target ) ) ;
2023-10-10 13:03:20 +02:00
}
return this . compareAvailableForValues ( trader . loyaltyLevel , requiredLoyaltyLevel , questProperties . compareMethod ) ;
}
/ * *
* Check if trader has sufficient standing to fulfill quest requirement
* @param questProperties Quest props
* @param profile Player profile
* @returns true if standing is high enough to fulfill quest requirement
* /
2024-01-05 20:52:21 +01:00
public traderStandingRequirementCheck ( questProperties : IQuestCondition , profile : IPmcData ) : boolean
2023-03-03 16:23:46 +01:00
{
2023-10-10 13:03:20 +02:00
const requiredStanding = Number ( questProperties . value ) ;
const trader = profile . TradersInfo [ < string > questProperties . target ] ;
if ( ! trader )
{
2024-05-21 13:40:16 +02:00
this . localisationService . getText ( "quest-unable_to_find_trader_in_profile" , questProperties . target ) ;
2023-10-10 13:03:20 +02:00
}
return this . compareAvailableForValues ( trader . standing , requiredStanding , questProperties . compareMethod ) ;
}
2023-03-03 16:23:46 +01:00
2023-10-10 13:03:20 +02:00
protected compareAvailableForValues ( current : number , required : number , compareMethod : string ) : boolean
{
switch ( compareMethod )
2023-03-03 16:23:46 +01:00
{
case ">=" :
2023-10-10 13:03:20 +02:00
return current >= required ;
2023-03-03 16:23:46 +01:00
case ">" :
2023-10-10 13:03:20 +02:00
return current > required ;
2023-03-03 16:23:46 +01:00
case "<=" :
2023-10-10 13:03:20 +02:00
return current <= required ;
2023-03-03 16:23:46 +01:00
case "<" :
2023-10-10 13:03:20 +02:00
return current < required ;
2023-03-03 16:23:46 +01:00
case "!=" :
2023-10-10 13:03:20 +02:00
return current !== required ;
2023-03-03 16:23:46 +01:00
case "==" :
2023-10-10 13:03:20 +02:00
return current === required ;
2023-11-16 22:42:06 +01:00
2023-03-03 16:23:46 +01:00
default :
2023-10-10 13:03:20 +02:00
this . logger . error ( this . localisationService . getText ( "quest-compare_operator_unhandled" , compareMethod ) ) ;
2023-03-03 16:23:46 +01:00
return false ;
}
}
2023-03-21 15:19:49 +01:00
/ * *
2024-01-05 20:52:21 +01:00
* Take reward item from quest and set FiR status + fix stack sizes + fix mod Ids
* @param questReward Reward item to fix
2023-03-21 15:19:49 +01:00
* @returns Fixed rewards
* /
2024-01-05 20:52:21 +01:00
protected processReward ( questReward : IQuestReward ) : Item [ ]
2023-03-03 16:23:46 +01:00
{
2024-01-10 11:04:09 +01:00
/** item with mods to return */
2024-01-05 20:52:21 +01:00
let rewardItems : Item [ ] = [ ] ;
2023-03-03 16:23:46 +01:00
let targets : Item [ ] = [ ] ;
const mods : Item [ ] = [ ] ;
2024-01-09 17:49:59 +01:00
2024-01-11 18:42:58 +01:00
// Is armor item that may need inserts / plates
2024-02-03 12:41:30 +01:00
if ( questReward . items . length === 1 && this . itemHelper . armorItemCanHoldMods ( questReward . items [ 0 ] . _tpl ) )
2024-01-09 17:49:59 +01:00
{
2024-02-03 12:00:30 +01:00
// Only process items with slots
2024-02-03 12:41:30 +01:00
if ( this . itemHelper . itemHasSlots ( questReward . items [ 0 ] . _tpl ) )
2024-02-03 12:00:30 +01:00
{
2024-02-03 13:15:20 +01:00
// Attempt to pull default preset from globals and add child items to reward (clones questReward.items)
2024-02-03 12:41:30 +01:00
this . generateArmorRewardChildSlots ( questReward . items [ 0 ] , questReward ) ;
2024-02-03 12:00:30 +01:00
}
2024-01-09 17:49:59 +01:00
}
2023-03-03 16:23:46 +01:00
2024-03-07 10:18:39 +01:00
for ( const rewardItem of questReward . items )
2023-03-03 16:23:46 +01:00
{
2024-03-07 10:18:39 +01:00
this . itemHelper . addUpdObjectToItem ( rewardItem ) ;
2024-02-03 13:15:20 +01:00
// Reward items are granted Found in Raid status
2024-03-07 10:18:39 +01:00
rewardItem . upd . SpawnedInSession = true ;
2023-03-03 16:23:46 +01:00
2024-02-03 12:41:30 +01:00
// Is root item, fix stacks
2024-03-07 10:18:39 +01:00
if ( rewardItem . _id === questReward . target )
2024-05-17 21:32:41 +02:00
{
// Is base reward item
2023-11-16 22:42:06 +01:00
if (
2024-05-17 21:32:41 +02:00
rewardItem . parentId !== undefined
&& rewardItem . parentId === "hideout" // Has parentId of hideout
&& rewardItem . upd !== undefined
&& rewardItem . upd . StackObjectsCount !== undefined // Has upd with stackobject count
2024-05-08 05:57:08 +02:00
&& rewardItem . upd . StackObjectsCount > 1 // More than 1 item in stack
2023-11-16 22:42:06 +01:00
)
2023-03-03 16:23:46 +01:00
{
2024-03-07 10:18:39 +01:00
rewardItem . upd . StackObjectsCount = 1 ;
2023-03-03 16:23:46 +01:00
}
2024-03-07 10:18:39 +01:00
targets = this . itemHelper . splitStack ( rewardItem ) ;
2023-03-03 16:23:46 +01:00
// splitStack created new ids for the new stacks. This would destroy the relation to possible children.
// Instead, we reset the id to preserve relations and generate a new id in the downstream loop, where we are also reparenting if required
for ( const target of targets )
{
2024-03-07 10:18:39 +01:00
target . _id = rewardItem . _id ;
2023-03-03 16:23:46 +01:00
}
}
else
{
2024-01-10 11:04:09 +01:00
// Is child mod
2024-02-03 12:41:30 +01:00
if ( questReward . items [ 0 ] . upd . SpawnedInSession )
2024-01-23 16:24:02 +01:00
{
// Propigate FiR status into child items
2024-03-07 10:18:39 +01:00
rewardItem . upd . SpawnedInSession = questReward . items [ 0 ] . upd . SpawnedInSession ;
2024-01-23 16:24:02 +01:00
}
2024-03-07 10:18:39 +01:00
mods . push ( rewardItem ) ;
2023-03-03 16:23:46 +01:00
}
}
// Add mods to the base items, fix ids
for ( const target of targets )
{
// This has all the original id relations since we reset the id to the original after the splitStack
2024-05-13 19:58:17 +02:00
const itemsClone = [ this . cloner . clone ( target ) ] ;
2023-03-03 16:23:46 +01:00
// Here we generate a new id for the root item
target . _id = this . hashUtil . generate ( ) ;
for ( const mod of mods )
{
2024-05-13 19:58:17 +02:00
itemsClone . push ( this . cloner . clone ( mod ) ) ;
2023-03-03 16:23:46 +01:00
}
2024-02-05 15:43:46 +01:00
rewardItems = rewardItems . concat ( this . itemHelper . reparentItemAndChildren ( target , itemsClone ) ) ;
2023-03-03 16:23:46 +01:00
}
return rewardItems ;
}
2024-01-10 11:04:09 +01:00
/ * *
* Add missing mod items to a quest armor reward
* @param originalRewardRootItem Original armor reward item from IQuestReward . items object
* @param questReward Armor reward from quest
* /
protected generateArmorRewardChildSlots ( originalRewardRootItem : Item , questReward : IQuestReward ) : void
{
// Look for a default preset from globals for armor
const defaultPreset = this . presetHelper . getDefaultPreset ( originalRewardRootItem . _tpl ) ;
if ( defaultPreset )
{
2024-02-03 13:47:52 +01:00
// Found preset, use mods to hydrate reward item
2024-02-06 04:22:03 +01:00
const presetAndMods : Item [ ] = this . itemHelper . replaceIDs ( defaultPreset . _items ) ;
2024-02-03 13:15:20 +01:00
const newRootId = this . itemHelper . remapRootItemId ( presetAndMods ) ;
questReward . items = presetAndMods ;
2024-01-10 11:04:09 +01:00
2024-02-03 13:47:52 +01:00
// Find root item and set its stack count
2024-05-17 21:32:41 +02:00
const rootItem = questReward . items . find ( ( item ) = > item . _id === newRootId ) ;
2024-02-03 13:47:52 +01:00
// Remap target id to the new presets root id
questReward . target = rootItem . _id ;
2024-02-03 15:00:47 +01:00
// Copy over stack count otherwise reward shows as missing in client
2024-03-07 10:18:39 +01:00
this . itemHelper . addUpdObjectToItem ( rootItem ) ;
2024-02-03 13:47:52 +01:00
rootItem . upd . StackObjectsCount = originalRewardRootItem . upd . StackObjectsCount ;
2024-01-10 11:04:09 +01:00
return ;
}
2024-02-02 19:54:07 +01:00
this . logger . warning (
` Unable to find default preset for armor ${ originalRewardRootItem . _tpl } , adding mods manually ` ,
) ;
2024-01-10 11:04:09 +01:00
const itemDbData = this . itemHelper . getItem ( originalRewardRootItem . _tpl ) [ 1 ] ;
// Hydrate reward with only 'required' mods - necessary for things like helmets otherwise you end up with nvgs/visors etc
2024-05-27 22:06:07 +02:00
questReward . items = this . itemHelper . addChildSlotItems ( questReward . items , itemDbData , undefined , true ) ;
2024-01-10 11:04:09 +01:00
}
2023-03-03 16:23:46 +01:00
/ * *
* Gets a flat list of reward items for the given quest at a specific state ( e . g . Fail / Success )
* @param quest quest to get rewards for
2023-03-21 15:19:49 +01:00
* @param status Quest status that holds the items ( Started , Success , Fail )
2023-03-03 16:23:46 +01:00
* @returns array of items with the correct maxStack
* /
2024-01-05 20:52:21 +01:00
public getQuestRewardItems ( quest : IQuest , status : QuestStatus ) : Item [ ]
2023-03-03 16:23:46 +01:00
{
2023-03-21 15:19:49 +01:00
// Iterate over all rewards with the desired status, flatten out items that have a type of Item
2024-01-05 20:52:21 +01:00
const questRewards = quest . rewards [ QuestStatus [ status ] ] . flatMap ( ( reward : IQuestReward ) = >
2024-05-17 21:32:41 +02:00
reward . type === "Item" ? this . processReward ( reward ) : [ ] ,
2023-11-16 22:42:06 +01:00
) ;
2023-03-03 16:23:46 +01:00
return questRewards ;
}
/ * *
* Look up quest in db by accepted quest id and construct a profile - ready object ready to store in profile
* @param pmcData Player profile
* @param newState State the new quest should be in when returned
* @param acceptedQuest Details of accepted quest from client
* /
2023-11-16 22:42:06 +01:00
public getQuestReadyForProfile (
pmcData : IPmcData ,
newState : QuestStatus ,
acceptedQuest : IAcceptQuestRequestData ,
) : IQuestStatus
2023-03-03 16:23:46 +01:00
{
2023-11-30 11:09:23 +01:00
const currentTimestamp = this . timeUtil . getTimestamp ( ) ;
2024-05-17 21:32:41 +02:00
const existingQuest = pmcData . Quests . find ( ( q ) = > q . qid === acceptedQuest . qid ) ;
2023-03-03 16:23:46 +01:00
if ( existingQuest )
{
// Quest exists, update its status
2023-11-30 11:09:23 +01:00
existingQuest . startTime = currentTimestamp ;
2023-03-03 16:23:46 +01:00
existingQuest . status = newState ;
2023-11-30 11:09:23 +01:00
existingQuest . statusTimers [ newState ] = currentTimestamp ;
2023-10-10 13:03:20 +02:00
existingQuest . completedConditions = [ ] ;
2023-03-03 16:23:46 +01:00
2023-10-10 17:18:55 +02:00
if ( existingQuest . availableAfter )
{
delete existingQuest . availableAfter ;
}
2023-03-03 16:23:46 +01:00
return existingQuest ;
}
// Quest doesn't exists, add it
2023-10-10 13:03:20 +02:00
const newQuest : IQuestStatus = {
2023-03-03 16:23:46 +01:00
qid : acceptedQuest.qid ,
2023-11-30 11:09:23 +01:00
startTime : currentTimestamp ,
2023-03-03 16:23:46 +01:00
status : newState ,
2023-11-16 22:42:06 +01:00
statusTimers : { } ,
2023-03-03 16:23:46 +01:00
} ;
2023-11-16 22:42:06 +01:00
2024-01-14 11:27:26 +01:00
// Check if quest has a prereq to be placed in a 'pending' state, otherwise set status timers value
2023-03-03 16:23:46 +01:00
const questDbData = this . getQuestFromDb ( acceptedQuest . qid , pmcData ) ;
2024-01-14 11:27:26 +01:00
if ( ! questDbData )
{
2024-05-21 13:40:16 +02:00
this . logger . error ( this . localisationService . getText ( "quest-unable_to_find_quest_in_db" , { questId : acceptedQuest.qid , questType : acceptedQuest.type } ) ) ;
2024-01-14 11:27:26 +01:00
}
2024-05-17 21:32:41 +02:00
const waitTime = questDbData ? . conditions . AvailableForStart . find ( ( x ) = > x . availableAfter > 0 ) ;
2023-03-03 16:23:46 +01:00
if ( waitTime && acceptedQuest . type !== "repeatable" )
{
// Quest should be put into 'pending' state
newQuest . startTime = 0 ;
newQuest . status = QuestStatus . AvailableAfter ; // 9
2023-12-27 18:15:38 +01:00
newQuest . availableAfter = currentTimestamp + waitTime . availableAfter ;
2023-03-03 16:23:46 +01:00
}
else
{
2023-11-30 11:09:23 +01:00
newQuest . statusTimers [ newState . toString ( ) ] = currentTimestamp ;
2023-03-03 16:23:46 +01:00
newQuest . completedConditions = [ ] ;
}
return newQuest ;
}
/ * *
2023-03-21 15:19:49 +01:00
* Get quests that can be shown to player after starting a quest
* @param startedQuestId Quest started by player
2023-03-03 16:23:46 +01:00
* @param sessionID Session id
2023-03-21 15:19:49 +01:00
* @returns Quests accessible to player incuding newly unlocked quests now quest ( startedQuestId ) was started
2023-03-03 16:23:46 +01:00
* /
2023-10-10 13:03:20 +02:00
public getNewlyAccessibleQuestsWhenStartingQuest ( startedQuestId : string , sessionID : string ) : IQuest [ ]
2023-03-03 16:23:46 +01:00
{
2023-03-21 15:19:49 +01:00
// Get quest acceptance data from profile
2023-10-10 13:03:20 +02:00
const profile : IPmcData = this . profileHelper . getPmcProfile ( sessionID ) ;
2024-05-17 21:32:41 +02:00
const startedQuestInProfile = profile . Quests . find ( ( profileQuest ) = > profileQuest . qid === startedQuestId ) ;
2023-03-03 16:23:46 +01:00
2023-11-16 22:42:06 +01:00
// Get quests that
2023-03-21 15:19:49 +01:00
const eligibleQuests = this . getQuestsFromDb ( ) . filter ( ( quest ) = >
2023-03-03 16:23:46 +01:00
{
2023-03-21 15:19:49 +01:00
// Quest is accessible to player when the accepted quest passed into param is started
// e.g. Quest A passed in, quest B is looped over and has requirement of A to be started, include it
2023-11-16 22:42:06 +01:00
const acceptedQuestCondition = quest . conditions . AvailableForStart . find ( ( x ) = >
2023-03-21 15:19:49 +01:00
{
2024-05-17 21:32:41 +02:00
return (
x . conditionType === "Quest"
&& x . target ? . includes ( startedQuestId )
&& x . status ? . includes ( QuestStatus . Started )
) ;
2023-03-21 15:19:49 +01:00
} ) ;
2023-03-03 16:23:46 +01:00
2023-03-21 15:19:49 +01:00
// Not found, skip quest
2023-03-03 16:23:46 +01:00
if ( ! acceptedQuestCondition )
{
return false ;
}
2024-03-24 18:17:40 +01:00
// Skip quest if its flagged as for other side
if ( this . questIsForOtherSide ( profile . Info . Side , quest . _id ) )
{
return false ;
}
2023-11-16 22:42:06 +01:00
const standingRequirements = this . questConditionHelper . getStandingConditions (
quest . conditions . AvailableForStart ,
) ;
2023-10-10 13:03:20 +02:00
for ( const condition of standingRequirements )
{
2023-12-27 18:15:38 +01:00
if ( ! this . traderStandingRequirementCheck ( condition , profile ) )
2023-10-10 13:03:20 +02:00
{
return false ;
}
}
2023-11-16 22:42:06 +01:00
const loyaltyRequirements = this . questConditionHelper . getLoyaltyConditions (
quest . conditions . AvailableForStart ,
) ;
2023-10-10 13:03:20 +02:00
for ( const condition of loyaltyRequirements )
{
2023-12-27 18:15:38 +01:00
if ( ! this . traderLoyaltyLevelRequirementCheck ( condition , profile ) )
2023-10-10 13:03:20 +02:00
{
return false ;
}
}
2023-03-21 15:19:49 +01:00
// Include if quest found in profile and is started or ready to hand in
2024-05-17 21:32:41 +02:00
return (
startedQuestInProfile
&& [ QuestStatus . Started , QuestStatus . AvailableForFinish ] . includes ( startedQuestInProfile . status )
) ;
2023-03-03 16:23:46 +01:00
} ) ;
2023-03-21 15:19:49 +01:00
return this . getQuestsWithOnlyLevelRequirementStartCondition ( eligibleQuests ) ;
2023-03-03 16:23:46 +01:00
}
2024-03-24 18:17:40 +01:00
/ * *
* Is the quest for the opposite side the player is on
* @param playerSide Player side ( usec / bear )
* @param questId QuestId to check
* /
public questIsForOtherSide ( playerSide : string , questId : string ) : boolean
{
const isUsec = playerSide . toLowerCase ( ) === "usec" ;
if ( isUsec && this . questConfig . bearOnlyQuests . includes ( questId ) )
{
// player is usec and quest is bear only, skip
return true ;
}
if ( ! isUsec && this . questConfig . usecOnlyQuests . includes ( questId ) )
{
// player is bear and quest is usec only, skip
return true ;
}
return false ;
}
2023-03-03 16:23:46 +01:00
/ * *
2023-03-21 15:19:49 +01:00
* Get quests that can be shown to player after failing a quest
* @param failedQuestId Id of the quest failed by player
* @param sessionId Session id
2023-07-25 20:50:46 +02:00
* @returns IQuest array
2023-03-03 16:23:46 +01:00
* /
2023-03-21 15:19:49 +01:00
public failedUnlocked ( failedQuestId : string , sessionId : string ) : IQuest [ ]
2023-03-03 16:23:46 +01:00
{
2023-03-21 15:19:49 +01:00
const profile = this . profileHelper . getPmcProfile ( sessionId ) ;
2024-05-17 21:32:41 +02:00
const profileQuest = profile . Quests . find ( ( x ) = > x . qid === failedQuestId ) ;
2023-03-03 16:23:46 +01:00
const quests = this . getQuestsFromDb ( ) . filter ( ( q ) = >
{
2023-11-16 22:42:06 +01:00
const acceptedQuestCondition = q . conditions . AvailableForStart . find ( ( c ) = >
{
2024-05-17 21:32:41 +02:00
return (
c . conditionType === "Quest" && c . target . includes ( failedQuestId ) && c . status [ 0 ] === QuestStatus . Fail
) ;
2023-11-16 22:42:06 +01:00
} ) ;
2023-03-03 16:23:46 +01:00
if ( ! acceptedQuestCondition )
{
return false ;
}
2024-05-08 05:57:08 +02:00
return profileQuest && profileQuest . status === QuestStatus . Fail ;
2023-03-03 16:23:46 +01:00
} ) ;
2023-07-25 20:50:46 +02:00
if ( quests . length === 0 )
{
return quests ;
}
2023-03-03 16:23:46 +01:00
return this . getQuestsWithOnlyLevelRequirementStartCondition ( quests ) ;
}
/ * *
* Adjust quest money rewards by passed in multiplier
* @param quest Quest to multiple money rewards
* @param multiplier Value to adjust money rewards by
2023-10-10 13:03:20 +02:00
* @param questStatus Status of quest to apply money boost to rewards of
2023-03-03 16:23:46 +01:00
* @returns Updated quest
* /
2023-10-10 13:03:20 +02:00
public applyMoneyBoost ( quest : IQuest , multiplier : number , questStatus : QuestStatus ) : IQuest
2023-03-03 16:23:46 +01:00
{
2024-01-05 20:52:21 +01:00
const rewards : IQuestReward [ ] = quest . rewards ? . [ QuestStatus [ questStatus ] ] ? ? [ ] ;
2023-10-10 13:03:20 +02:00
for ( const reward of rewards )
2023-03-03 16:23:46 +01:00
{
if ( reward . type === "Item" )
{
if ( this . paymentHelper . isMoneyTpl ( reward . items [ 0 ] . _tpl ) )
{
2023-11-16 22:42:06 +01:00
reward . items [ 0 ] . upd . StackObjectsCount += Math . round (
2024-05-17 21:32:41 +02:00
( reward . items [ 0 ] . upd . StackObjectsCount * multiplier ) / 100 ,
2023-11-16 22:42:06 +01:00
) ;
2023-03-03 16:23:46 +01:00
}
}
}
return quest ;
}
/ * *
* Sets the item stack to new value , or delete the item if value <= 0
* // TODO maybe merge this function and the one from customization
* @param pmcData Profile
* @param itemId id of item to adjust stack size of
* @param newStackSize Stack size to adjust to
* @param sessionID Session id
* @param output ItemEvent router response
* /
2023-11-16 22:42:06 +01:00
public changeItemStack (
pmcData : IPmcData ,
itemId : string ,
newStackSize : number ,
sessionID : string ,
output : IItemEventRouterResponse ,
) : void
2023-03-03 16:23:46 +01:00
{
2024-05-17 21:32:41 +02:00
const inventoryItemIndex = pmcData . Inventory . items . findIndex ( ( item ) = > item . _id === itemId ) ;
2023-03-03 16:23:46 +01:00
if ( inventoryItemIndex < 0 )
{
this . logger . error ( this . localisationService . getText ( "quest-item_not_found_in_inventory" , itemId ) ) ;
return ;
}
if ( newStackSize > 0 )
{
const item = pmcData . Inventory . items [ inventoryItemIndex ] ;
2024-03-07 10:18:39 +01:00
this . itemHelper . addUpdObjectToItem ( item ) ;
2023-03-03 16:23:46 +01:00
item . upd . StackObjectsCount = newStackSize ;
2023-03-21 15:19:49 +01:00
this . addItemStackSizeChangeIntoEventResponse ( output , sessionID , item ) ;
2023-03-03 16:23:46 +01:00
}
else
{
// this case is probably dead Code right now, since the only calling function
// checks explicitly for Value > 0.
2023-11-16 22:42:06 +01:00
output . profileChanges [ sessionID ] . items . del . push ( { _id : itemId } ) ;
2023-03-03 16:23:46 +01:00
pmcData . Inventory . items . splice ( inventoryItemIndex , 1 ) ;
}
}
2023-03-21 15:19:49 +01:00
/ * *
* Add item stack change object into output route event response
* @param output Response to add item change event into
* @param sessionId Session id
* @param item Item that was adjusted
* /
2023-11-16 22:42:06 +01:00
protected addItemStackSizeChangeIntoEventResponse (
output : IItemEventRouterResponse ,
sessionId : string ,
item : Item ,
) : void
2023-03-21 15:19:49 +01:00
{
output . profileChanges [ sessionId ] . items . change . push ( {
2023-11-16 22:42:06 +01:00
_id : item._id ,
_tpl : item._tpl ,
parentId : item.parentId ,
slotId : item.slotId ,
location : item.location ,
upd : { StackObjectsCount : item.upd.StackObjectsCount } ,
2023-03-21 15:19:49 +01:00
} ) ;
}
2023-03-03 16:23:46 +01:00
/ * *
* Get quests , strip all requirement conditions except level
* @param quests quests to process
* @returns quest array without conditions
* /
protected getQuestsWithOnlyLevelRequirementStartCondition ( quests : IQuest [ ] ) : IQuest [ ]
{
for ( const i in quests )
{
quests [ i ] = this . getQuestWithOnlyLevelRequirementStartCondition ( quests [ i ] ) ;
}
return quests ;
}
/ * *
* Remove all quest conditions except for level requirement
* @param quest quest to clean
* @returns reset IQuest object
* /
public getQuestWithOnlyLevelRequirementStartCondition ( quest : IQuest ) : IQuest
{
2024-05-13 19:58:17 +02:00
const updatedQuest = this . cloner . clone ( quest ) ;
2024-05-17 21:32:41 +02:00
updatedQuest . conditions . AvailableForStart = updatedQuest . conditions . AvailableForStart . filter (
( q ) = > q . conditionType === "Level" ,
2024-02-02 19:54:07 +01:00
) ;
2023-03-03 16:23:46 +01:00
2024-04-23 05:43:35 +02:00
return updatedQuest ;
2023-03-03 16:23:46 +01:00
}
/ * *
* Fail a quest in a player profile
* @param pmcData Player profile
* @param failRequest Fail quest request data
* @param sessionID Session id
2023-10-10 13:03:20 +02:00
* @param output Client output
2023-03-03 16:23:46 +01:00
* /
2023-11-16 22:42:06 +01:00
public failQuest (
pmcData : IPmcData ,
failRequest : IFailQuestRequestData ,
sessionID : string ,
2024-05-27 22:06:07 +02:00
output? : IItemEventRouterResponse ,
2024-03-12 22:29:50 +01:00
) : void
2023-03-03 16:23:46 +01:00
{
2024-04-23 05:43:35 +02:00
let updatedOutput = output ;
2024-03-12 22:24:46 +01:00
// Prepare response to send back to client
2024-04-23 05:43:35 +02:00
if ( ! updatedOutput )
2023-10-10 13:03:20 +02:00
{
2024-04-23 05:43:35 +02:00
updatedOutput = this . eventOutputHolder . getOutput ( sessionID ) ;
2023-10-10 13:03:20 +02:00
}
2023-03-03 16:23:46 +01:00
this . updateQuestState ( pmcData , QuestStatus . Fail , failRequest . qid ) ;
2024-04-23 05:43:35 +02:00
const questRewards = this . applyQuestReward (
pmcData ,
failRequest . qid ,
QuestStatus . Fail ,
sessionID ,
updatedOutput ,
) ;
2023-03-03 16:23:46 +01:00
// Create a dialog message for completing the quest.
const quest = this . getQuestFromDb ( failRequest . qid , pmcData ) ;
2024-03-12 22:24:46 +01:00
// Merge all daily/weekly/scav daily quests into one array and look for the matching quest by id
2024-05-17 21:32:41 +02:00
const matchingRepeatableQuest = pmcData . RepeatableQuests . flatMap (
( repeatableType ) = > repeatableType . activeQuests ,
) . find ( ( activeQuest ) = > activeQuest . _id === failRequest . qid ) ;
2024-02-04 00:10:42 +01:00
2024-04-12 12:20:30 +02:00
// Quest found and no repeatable found
if ( quest && ! matchingRepeatableQuest )
2024-01-08 00:51:28 +01:00
{
2024-04-12 12:29:50 +02:00
if ( quest . failMessageText . trim ( ) . length > 0 )
{
this . mailSendService . sendLocalisedNpcMessageToPlayer (
sessionID ,
2024-05-27 22:06:07 +02:00
this . traderHelper . getTraderById ( quest ? . traderId ? ? matchingRepeatableQuest ? . traderId ) , // Can be undefined when repeatable quest has been moved to inactiveQuests
2024-04-12 12:29:50 +02:00
MessageType . QUEST_FAIL ,
quest . failMessageText ,
questRewards ,
2024-05-20 12:31:45 +02:00
this . timeUtil . getHoursAsSeconds ( this . getMailItemRedeemTimeHoursForProfile ( pmcData ) ) ,
2024-04-12 12:29:50 +02:00
) ;
}
2024-01-08 00:51:28 +01:00
}
2023-03-03 16:23:46 +01:00
2024-04-23 05:43:35 +02:00
updatedOutput . profileChanges [ sessionID ] . quests . push ( . . . this . failedUnlocked ( failRequest . qid , sessionID ) ) ;
2023-03-03 16:23:46 +01:00
}
/ * *
* Get List of All Quests from db
* NOT CLONED
* @returns Array of IQuest objects
* /
public getQuestsFromDb ( ) : IQuest [ ]
{
2024-05-29 16:15:45 +02:00
return Object . values ( this . databaseService . getQuests ( ) ) ;
2023-03-03 16:23:46 +01:00
}
/ * *
* Get quest by id from database ( repeatables are stored in profile , check there if questId not found )
* @param questId Id of quest to find
* @param pmcData Player profile
* @returns IQuest object
* /
public getQuestFromDb ( questId : string , pmcData : IPmcData ) : IQuest
{
// May be a repeatable quest
2024-05-29 16:15:45 +02:00
let quest = this . databaseService . getQuests ( ) [ questId ] ;
2023-03-03 16:23:46 +01:00
if ( ! quest )
{
// Check daily/weekly objects
for ( const repeatableType of pmcData . RepeatableQuests )
{
2024-05-29 16:15:45 +02:00
quest = < IQuest > ( < unknown > repeatableType . activeQuests . find ( ( repeatable ) = > repeatable . _id === questId ) ) ;
2023-03-03 16:23:46 +01:00
if ( quest )
{
break ;
}
}
}
return quest ;
}
/ * *
* Get a quests startedMessageText key from db , if no startedMessageText key found , use description key instead
* @param startedMessageTextId startedMessageText property from IQuest
* @param questDescriptionId description property from IQuest
* @returns message id
* /
public getMessageIdForQuestStart ( startedMessageTextId : string , questDescriptionId : string ) : string
{
// blank or is a guid, use description instead
const startedMessageText = this . getQuestLocaleIdFromDb ( startedMessageTextId ) ;
2023-11-16 22:42:06 +01:00
if (
2024-05-17 21:32:41 +02:00
! startedMessageText
|| startedMessageText . trim ( ) === ""
|| startedMessageText . toLowerCase ( ) === "test"
2023-11-16 22:42:06 +01:00
|| startedMessageText . length === 24
)
2023-03-03 16:23:46 +01:00
{
return questDescriptionId ;
}
return startedMessageTextId ;
}
/ * *
* Get the locale Id from locale db for a quest message
* @param questMessageId Quest message id to look up
* @returns Locale Id from locale db
* /
public getQuestLocaleIdFromDb ( questMessageId : string ) : string
{
const locale = this . localeService . getLocaleDb ( ) ;
return locale [ questMessageId ] ;
}
/ * *
* Alter a quests state + Add a record to its status timers object
* @param pmcData Profile to update
* @param newQuestState New state the quest should be in
* @param questId Id of the quest to alter the status of
* /
public updateQuestState ( pmcData : IPmcData , newQuestState : QuestStatus , questId : string ) : void
{
// Find quest in profile, update status to desired status
2024-05-17 21:32:41 +02:00
const questToUpdate = pmcData . Quests . find ( ( quest ) = > quest . qid === questId ) ;
2023-03-03 16:23:46 +01:00
if ( questToUpdate )
{
questToUpdate . status = newQuestState ;
questToUpdate . statusTimers [ newQuestState ] = this . timeUtil . getTimestamp ( ) ;
}
}
2023-11-30 00:36:31 +01:00
/ * *
* Resets a quests values back to its chosen state
* @param pmcData Profile to update
* @param newQuestState New state the quest should be in
* @param questId Id of the quest to alter the status of
* /
public resetQuestState ( pmcData : IPmcData , newQuestState : QuestStatus , questId : string ) : void
{
2024-05-17 21:32:41 +02:00
const questToUpdate = pmcData . Quests . find ( ( quest ) = > quest . qid === questId ) ;
2023-11-30 00:36:31 +01:00
if ( questToUpdate )
{
2023-11-30 11:09:23 +01:00
const currentTimestamp = this . timeUtil . getTimestamp ( ) ;
2023-11-30 00:36:31 +01:00
questToUpdate . status = newQuestState ;
2023-11-30 11:13:45 +01:00
// Only set start time when quest is being started
if ( newQuestState === QuestStatus . Started )
{
questToUpdate . startTime = currentTimestamp ;
}
2023-11-30 11:09:23 +01:00
questToUpdate . statusTimers [ newQuestState ] = currentTimestamp ;
2023-11-30 00:36:31 +01:00
2023-11-30 10:36:28 +01:00
// Delete all status timers after applying new status
2023-11-30 00:36:31 +01:00
for ( const statusKey in questToUpdate . statusTimers )
{
if ( Number . parseInt ( statusKey ) > newQuestState )
{
2023-11-30 11:09:23 +01:00
delete questToUpdate . statusTimers [ statusKey ] ;
2023-11-30 00:36:31 +01:00
}
}
// Remove all completed conditions
questToUpdate . completedConditions = [ ] ;
}
}
2023-03-03 16:23:46 +01:00
/ * *
* Give player quest rewards - Skills / exp / trader standing / items / assort unlocks - Returns reward items player earned
2023-11-20 12:19:11 +01:00
* @param profileData Player profile ( scav or pmc )
2023-03-03 16:23:46 +01:00
* @param questId questId of quest to get rewards for
* @param state State of the quest to get rewards for
* @param sessionId Session id
* @param questResponse Response to send back to client
* @returns Array of reward objects
* /
2023-11-16 22:42:06 +01:00
public applyQuestReward (
2023-11-20 12:19:11 +01:00
profileData : IPmcData ,
2023-11-16 22:42:06 +01:00
questId : string ,
state : QuestStatus ,
sessionId : string ,
questResponse : IItemEventRouterResponse ,
2024-01-05 20:52:21 +01:00
) : Item [ ]
2023-11-16 22:42:06 +01:00
{
2023-11-20 12:19:11 +01:00
// Repeatable quest base data is always in PMCProfile, `profileData` may be scav profile
// TODO: consider moving repeatable quest data to profile-agnostic location
const pmcProfile = this . profileHelper . getPmcProfile ( sessionId ) ;
let questDetails = this . getQuestFromDb ( questId , pmcProfile ) ;
2023-10-10 13:03:20 +02:00
if ( ! questDetails )
{
2024-05-21 15:28:52 +02:00
this . logger . warning ( this . localisationService . getText ( "quest-unable_to_find_quest_in_db_no_quest_rewards" , questId ) ) ;
2023-10-10 13:03:20 +02:00
return [ ] ;
}
2023-11-16 22:42:06 +01:00
2023-03-03 16:23:46 +01:00
// Check for and apply intel center money bonus if it exists
2023-11-20 12:19:11 +01:00
const questMoneyRewardBonus = this . getQuestMoneyRewardBonus ( pmcProfile ) ;
2023-10-10 13:03:20 +02:00
if ( questMoneyRewardBonus > 0 )
2023-03-03 16:23:46 +01:00
{
2023-10-10 13:03:20 +02:00
// Apply additional bonus from hideout skill
questDetails = this . applyMoneyBoost ( questDetails , questMoneyRewardBonus , state ) ; // money = money + (money * intelCenterBonus / 100)
2023-03-03 16:23:46 +01:00
}
// e.g. 'Success' or 'AvailableForFinish'
const questStateAsString = QuestStatus [ state ] ;
2024-01-05 20:52:21 +01:00
for ( const reward of < IQuestReward [ ] > questDetails . rewards [ questStateAsString ] )
2023-03-03 16:23:46 +01:00
{
switch ( reward . type )
{
case QuestRewardType . SKILL :
2023-11-16 22:42:06 +01:00
this . profileHelper . addSkillPointsToPlayer (
2023-11-20 12:19:11 +01:00
profileData ,
2023-11-16 22:42:06 +01:00
reward . target as SkillTypes ,
Number ( reward . value ) ,
) ;
2023-03-03 16:23:46 +01:00
break ;
case QuestRewardType . EXPERIENCE :
2024-03-24 18:17:40 +01:00
this . profileHelper . addExperienceToPmc ( sessionId , Number . parseInt ( < string > reward . value ) ) ; // this must occur first as the output object needs to take the modified profile exp value
2023-03-03 16:23:46 +01:00
break ;
case QuestRewardType . TRADER_STANDING :
2024-03-24 18:17:40 +01:00
this . traderHelper . addStandingToTrader (
sessionId ,
reward . target ,
Number . parseFloat ( < string > reward . value ) ,
) ;
2023-03-03 16:23:46 +01:00
break ;
case QuestRewardType . TRADER_UNLOCK :
this . traderHelper . setTraderUnlockedState ( reward . target , true , sessionId ) ;
break ;
case QuestRewardType . ITEM :
// Handled by getQuestRewardItems() below
break ;
case QuestRewardType . ASSORTMENT_UNLOCK :
// Handled elsewhere, TODO: find and say here
break ;
2024-05-30 14:05:28 +02:00
case QuestRewardType . ACHIEVEMENT :
2024-06-04 16:36:01 +02:00
this . profileHelper . addAchievementToProfile ( pmcProfile , reward . target ) ;
2024-05-30 14:05:28 +02:00
break ;
2023-08-06 17:54:31 +02:00
case QuestRewardType . STASH_ROWS :
2024-05-01 10:22:11 +02:00
this . profileHelper . addStashRowsBonusToProfile ( sessionId , Number . parseInt ( < string > reward . value ) ) ; // add specified stash rows from quest reward - requires client restart
2023-08-06 17:54:31 +02:00
break ;
2023-03-03 16:23:46 +01:00
case QuestRewardType . PRODUCTIONS_SCHEME :
2023-11-16 22:42:06 +01:00
this . findAndAddHideoutProductionIdToProfile (
2023-11-20 12:19:11 +01:00
pmcProfile ,
2023-11-16 22:42:06 +01:00
reward ,
questDetails ,
sessionId ,
questResponse ,
) ;
2023-03-03 16:23:46 +01:00
break ;
default :
2023-11-16 22:42:06 +01:00
this . logger . error (
this . localisationService . getText ( "quest-reward_type_not_handled" , {
rewardType : reward.type ,
questId : questId ,
questName : questDetails.QuestName ,
} ) ,
) ;
2023-03-03 16:23:46 +01:00
break ;
}
}
return this . getQuestRewardItems ( questDetails , state ) ;
}
/ * *
* WIP - Find hideout craft id and add to unlockedProductionRecipe array in player profile
* also update client response recipeUnlocked array with craft id
* @param pmcData Player profile
* @param craftUnlockReward Reward item from quest with craft unlock details
* @param questDetails Quest with craft unlock reward
* @param sessionID Session id
* @param response Response to send back to client
* /
2023-11-16 22:42:06 +01:00
protected findAndAddHideoutProductionIdToProfile (
pmcData : IPmcData ,
2024-01-05 20:52:21 +01:00
craftUnlockReward : IQuestReward ,
2023-11-16 22:42:06 +01:00
questDetails : IQuest ,
sessionID : string ,
response : IItemEventRouterResponse ,
) : void
2023-03-03 16:23:46 +01:00
{
// Get hideout crafts and find those that match by areatype/required level/end product tpl - hope for just one match
2024-05-29 16:15:45 +02:00
const hideoutProductions = this . databaseService . getHideout ( ) . production ;
2024-05-17 21:32:41 +02:00
const matchingProductions = hideoutProductions . filter (
2024-05-29 16:15:45 +02:00
( prod ) = >
prod . areaType === Number . parseInt ( craftUnlockReward . traderId )
&& prod . requirements . some ( ( x ) = > x . requiredLevel === craftUnlockReward . loyaltyLevel )
2024-06-06 12:34:03 +02:00
&& prod . endProduct === craftUnlockReward . items [ 0 ] . _tpl
&& prod . count === ( craftUnlockReward . items [ 0 ] . upd ? . StackObjectsCount ? ? 1 ) ,
2023-11-16 22:42:06 +01:00
) ;
2023-03-03 16:23:46 +01:00
2023-08-02 13:57:08 +02:00
// More/less than 1 match, above filtering wasn't strict enough
2023-03-03 16:23:46 +01:00
if ( matchingProductions . length !== 1 )
{
2023-11-16 22:42:06 +01:00
this . logger . error (
this . localisationService . getText ( "quest-unable_to_find_matching_hideout_production" , {
questName : questDetails.QuestName ,
matchCount : matchingProductions.length ,
} ) ,
) ;
2023-03-03 16:23:46 +01:00
return ;
}
// Add above match to pmc profile + client response
const matchingCraftId = matchingProductions [ 0 ] . _id ;
pmcData . UnlockedInfo . unlockedProductionRecipe . push ( matchingCraftId ) ;
response . profileChanges [ sessionID ] . recipeUnlocked [ matchingCraftId ] = true ;
}
/ * *
2023-10-10 13:03:20 +02:00
* Get players money reward bonus from profile
2023-03-03 16:23:46 +01:00
* @param pmcData player profile
* @returns bonus as a percent
* /
2023-10-10 13:03:20 +02:00
protected getQuestMoneyRewardBonus ( pmcData : IPmcData ) : number
2023-03-03 16:23:46 +01:00
{
// Check player has intel center
2024-05-17 21:32:41 +02:00
const moneyRewardBonuses = pmcData . Bonuses . filter ( ( x ) = > x . type === "QuestMoneyReward" ) ;
2023-10-10 13:03:20 +02:00
if ( ! moneyRewardBonuses )
2023-03-03 16:23:46 +01:00
{
2023-10-10 13:03:20 +02:00
return 0 ;
}
2023-03-03 16:23:46 +01:00
2023-10-10 13:03:20 +02:00
// Get a total of the quest money rewards
let moneyRewardBonus = moneyRewardBonuses . reduce ( ( acc , cur ) = > acc + cur . value , 0 ) ;
// Apply hideout management bonus to money reward (up to 51% bonus)
2023-11-07 16:17:38 +01:00
const hideoutManagementSkill = this . profileHelper . getSkillFromProfile ( pmcData , SkillTypes . HIDEOUT_MANAGEMENT ) ;
2023-10-10 13:03:20 +02:00
if ( hideoutManagementSkill )
{
2024-05-08 05:57:08 +02:00
moneyRewardBonus *= 1 + hideoutManagementSkill . Progress / 10000 ; // 5100 becomes 0.51, add 1 to it, 1.51, multiply the moneyreward bonus by it (e.g. 15 x 51)
2023-03-03 16:23:46 +01:00
}
2023-10-10 13:03:20 +02:00
return moneyRewardBonus ;
2023-03-03 16:23:46 +01:00
}
/ * *
2023-10-10 13:03:20 +02:00
* Find quest with 'findItem' condition that needs the item tpl be handed in
2023-03-03 16:23:46 +01:00
* @param itemTpl item tpl to look for
2023-10-10 13:03:20 +02:00
* @param questIds Quests to search through for the findItem condition
* @returns quest id with 'FindItem' condition id
2023-03-03 16:23:46 +01:00
* /
2023-11-16 22:42:06 +01:00
public getFindItemConditionByQuestItem (
itemTpl : string ,
questIds : string [ ] ,
allQuests : IQuest [ ] ,
) : Record < string , string >
2023-03-03 16:23:46 +01:00
{
2023-10-10 13:03:20 +02:00
const result : Record < string , string > = { } ;
for ( const questId of questIds )
2023-03-03 16:23:46 +01:00
{
2024-05-17 21:32:41 +02:00
const questInDb = allQuests . find ( ( x ) = > x . _id === questId ) ;
2023-10-10 13:03:20 +02:00
if ( ! questInDb )
{
2024-02-02 19:54:07 +01:00
this . logger . debug ( ` Unable to find quest: ${ questId } in db, cannot get 'FindItem' condition, skipping ` ) ;
2023-10-10 13:03:20 +02:00
continue ;
}
2024-05-17 21:32:41 +02:00
const condition = questInDb . conditions . AvailableForFinish . find (
( c ) = > c . conditionType === "FindItem" && c ? . target ? . includes ( itemTpl ) ,
2023-11-16 22:42:06 +01:00
) ;
2023-03-03 16:23:46 +01:00
if ( condition )
{
2023-12-27 18:15:38 +01:00
result [ questId ] = condition . id ;
2023-10-10 13:03:20 +02:00
break ;
2023-03-03 16:23:46 +01:00
}
}
2023-03-03 18:53:28 +01:00
return result ;
2023-03-03 16:23:46 +01:00
}
/ * *
* Add all quests to a profile with the provided statuses
* @param pmcProfile profile to update
* @param statuses statuses quests should have
* /
public addAllQuestsToProfile ( pmcProfile : IPmcData , statuses : QuestStatus [ ] ) : void
{
// Iterate over all quests in db
2024-05-29 16:15:45 +02:00
const quests = this . databaseService . getQuests ( ) ;
2024-02-03 13:47:52 +01:00
for ( const questIdKey in quests )
2023-03-03 16:23:46 +01:00
{
// Quest from db matches quests in profile, skip
2024-02-03 13:47:52 +01:00
const questData = quests [ questIdKey ] ;
2024-05-17 21:32:41 +02:00
if ( pmcProfile . Quests . find ( ( x ) = > x . qid === questData . _id ) )
2023-03-03 16:23:46 +01:00
{
continue ;
}
const statusesDict = { } ;
for ( const status of statuses )
{
statusesDict [ status ] = this . timeUtil . getTimestamp ( ) ;
}
2023-10-10 13:03:20 +02:00
const questRecordToAdd : IQuestStatus = {
2024-02-03 13:47:52 +01:00
qid : questIdKey ,
2023-10-10 13:03:20 +02:00
startTime : this.timeUtil.getTimestamp ( ) ,
2023-03-03 16:23:46 +01:00
status : statuses [ statuses . length - 1 ] ,
statusTimers : statusesDict ,
completedConditions : [ ] ,
2023-11-16 22:42:06 +01:00
availableAfter : 0 ,
2023-03-03 16:23:46 +01:00
} ;
2023-10-10 13:03:20 +02:00
2024-05-17 21:32:41 +02:00
if ( pmcProfile . Quests . some ( ( x ) = > x . qid === questIdKey ) )
2023-10-10 13:03:20 +02:00
{
// Update existing
2024-05-17 21:32:41 +02:00
const existingQuest = pmcProfile . Quests . find ( ( x ) = > x . qid === questIdKey ) ;
2023-10-10 13:03:20 +02:00
existingQuest . status = questRecordToAdd . status ;
existingQuest . statusTimers = questRecordToAdd . statusTimers ;
}
else
{
// Add new
pmcProfile . Quests . push ( questRecordToAdd ) ;
}
2023-10-21 18:39:44 +02:00
}
}
public findAndRemoveQuestFromArrayIfExists ( questId : string , quests : IQuestStatus [ ] ) : void
{
2024-05-17 21:32:41 +02:00
const pmcQuestToReplaceStatus = quests . find ( ( quest ) = > quest . qid === questId ) ;
2023-10-21 18:39:44 +02:00
if ( pmcQuestToReplaceStatus )
{
2024-02-10 00:18:23 +01:00
quests . splice ( quests . indexOf ( pmcQuestToReplaceStatus ) , 1 ) ;
2023-03-03 16:23:46 +01:00
}
}
2024-01-14 13:29:58 +01:00
/ * *
* Return a list of quests that would fail when supplied quest is completed
* @param completedQuestId quest completed id
* @returns array of IQuest objects
* /
public getQuestsFailedByCompletingQuest ( completedQuestId : string ) : IQuest [ ]
{
const questsInDb = this . getQuestsFromDb ( ) ;
return questsInDb . filter ( ( quest ) = >
{
// No fail conditions, exit early
if ( ! quest . conditions . Fail || quest . conditions . Fail . length === 0 )
{
return false ;
}
2024-05-17 21:32:41 +02:00
return quest . conditions . Fail . some ( ( condition ) = > condition . target ? . includes ( completedQuestId ) ) ;
2024-01-14 13:29:58 +01:00
} ) ;
}
2024-05-20 12:31:45 +02:00
/ * *
* Get the hours a mails items can be collected for by profile type
* @param pmcData Profile to get hours for
* @returns Hours item will be available for
* /
public getMailItemRedeemTimeHoursForProfile ( pmcData : IPmcData ) : number
{
const value = this . questConfig . mailRedeemTimeHours [ pmcData . Info . GameVersion ] ;
if ( ! value )
{
return this . questConfig . mailRedeemTimeHours [ "default" ] ;
}
return value ;
}
2023-11-16 22:42:06 +01:00
}