2023-03-03 15:23:46 +00:00
import { inject , injectable } from "tsyringe" ;
import { IPmcData } from "../models/eft/common/IPmcData" ;
import { Item } from "../models/eft/common/tables/IItem" ;
import { ITraderAssort } from "../models/eft/common/tables/ITrader" ;
import { IItemEventRouterResponse } from "../models/eft/itemEvent/IItemEventRouterResponse" ;
2023-05-29 16:12:27 +01:00
import { IAkiProfile , ISystemData } from "../models/eft/profile/IAkiProfile" ;
2023-03-03 15:23:46 +00:00
import { IRagfairOffer } from "../models/eft/ragfair/IRagfairOffer" ;
import { ISearchRequestData , OfferOwnerType } from "../models/eft/ragfair/ISearchRequestData" ;
import { ConfigTypes } from "../models/enums/ConfigTypes" ;
import { MemberCategory } from "../models/enums/MemberCategory" ;
import { MessageType } from "../models/enums/MessageType" ;
import { RagfairSort } from "../models/enums/RagfairSort" ;
import { Traders } from "../models/enums/Traders" ;
import { IQuestConfig } from "../models/spt/config/IQuestConfig" ;
import { IRagfairConfig } from "../models/spt/config/IRagfairConfig" ;
import { ILogger } from "../models/spt/utils/ILogger" ;
import { EventOutputHolder } from "../routers/EventOutputHolder" ;
import { ConfigServer } from "../servers/ConfigServer" ;
import { DatabaseServer } from "../servers/DatabaseServer" ;
import { SaveServer } from "../servers/SaveServer" ;
import { LocaleService } from "../services/LocaleService" ;
2023-07-19 11:00:34 +01:00
import { LocalisationService } from "../services/LocalisationService" ;
2023-03-03 15:23:46 +00:00
import { RagfairOfferService } from "../services/RagfairOfferService" ;
import { HashUtil } from "../utils/HashUtil" ;
import { TimeUtil } from "../utils/TimeUtil" ;
import { DialogueHelper } from "./DialogueHelper" ;
import { ItemHelper } from "./ItemHelper" ;
import { PaymentHelper } from "./PaymentHelper" ;
import { PresetHelper } from "./PresetHelper" ;
import { ProfileHelper } from "./ProfileHelper" ;
import { RagfairHelper } from "./RagfairHelper" ;
import { RagfairServerHelper } from "./RagfairServerHelper" ;
import { RagfairSortHelper } from "./RagfairSortHelper" ;
import { TraderHelper } from "./TraderHelper" ;
@injectable ( )
export class RagfairOfferHelper
{
2023-05-29 16:05:19 +01:00
protected static goodSoldTemplate = "5bdabfb886f7743e152e867e 0" ; // Your {soldItem} {itemCount} items were bought by {buyerNickname}.
2023-03-03 15:23:46 +00:00
protected ragfairConfig : IRagfairConfig ;
protected questConfig : IQuestConfig ;
constructor (
@inject ( "WinstonLogger" ) protected logger : ILogger ,
@inject ( "TimeUtil" ) protected timeUtil : TimeUtil ,
@inject ( "HashUtil" ) protected hashUtil : HashUtil ,
@inject ( "EventOutputHolder" ) protected eventOutputHolder : EventOutputHolder ,
@inject ( "DatabaseServer" ) protected databaseServer : DatabaseServer ,
@inject ( "TraderHelper" ) protected traderHelper : TraderHelper ,
@inject ( "SaveServer" ) protected saveServer : SaveServer ,
@inject ( "DialogueHelper" ) protected dialogueHelper : DialogueHelper ,
@inject ( "ItemHelper" ) protected itemHelper : ItemHelper ,
@inject ( "PaymentHelper" ) protected paymentHelper : PaymentHelper ,
@inject ( "PresetHelper" ) protected presetHelper : PresetHelper ,
@inject ( "ProfileHelper" ) protected profileHelper : ProfileHelper ,
@inject ( "RagfairServerHelper" ) protected ragfairServerHelper : RagfairServerHelper ,
@inject ( "RagfairSortHelper" ) protected ragfairSortHelper : RagfairSortHelper ,
@inject ( "RagfairHelper" ) protected ragfairHelper : RagfairHelper ,
@inject ( "RagfairOfferService" ) protected ragfairOfferService : RagfairOfferService ,
@inject ( "LocaleService" ) protected localeService : LocaleService ,
2023-07-19 11:00:34 +01:00
@inject ( "LocalisationService" ) protected localisationService : LocalisationService ,
2023-03-03 15:23:46 +00:00
@inject ( "ConfigServer" ) protected configServer : ConfigServer
)
{
this . ragfairConfig = this . configServer . getConfig ( ConfigTypes . RAGFAIR ) ;
this . questConfig = this . configServer . getConfig ( ConfigTypes . QUEST ) ;
}
2023-05-29 16:32:06 +01:00
/ * *
* Passthrough to ragfairOfferService . getOffers ( ) , get flea offers a player should see
* @param searchRequest
* @param itemsToAdd
* @param traderAssorts Trader assorts
* @param pmcProfile Player profile
* @returns Offers the player should see
* /
2023-03-03 15:23:46 +00:00
public getValidOffers ( searchRequest : ISearchRequestData , itemsToAdd : string [ ] , traderAssorts : Record < string , ITraderAssort > , pmcProfile : IPmcData ) : IRagfairOffer [ ]
{
return this . ragfairOfferService . getOffers ( ) . filter ( x = > this . isDisplayableOffer ( searchRequest , itemsToAdd , traderAssorts , x , pmcProfile ) ) ;
}
/ * *
* Get offers from flea / traders specifically when building weapon preset
* @param searchRequest Search request data
* @param itemsToAdd string array of item tpls to search for
* @param traderAssorts All trader assorts player can access / buy
* @param pmcProfile Player profile
2023-05-29 16:32:06 +01:00
* @returns IRagfairOffer array
2023-03-03 15:23:46 +00:00
* /
public getOffersForBuild ( searchRequest : ISearchRequestData , itemsToAdd : string [ ] , traderAssorts : Record < string , ITraderAssort > , pmcProfile : IPmcData ) : IRagfairOffer [ ]
{
const offersMap = new Map < string , IRagfairOffer [ ] > ( ) ;
const offers : IRagfairOffer [ ] = [ ] ;
for ( const offer of this . ragfairOfferService . getOffers ( ) )
{
if ( this . isDisplayableOffer ( searchRequest , itemsToAdd , traderAssorts , offer , pmcProfile ) )
{
const isTraderOffer = offer . user . memberType === MemberCategory . TRADER ;
if ( isTraderOffer && this . traderBuyRestrictionReached ( offer ) )
{
continue ;
}
if ( isTraderOffer && this . traderOutOfStock ( offer ) )
{
continue ;
}
if ( isTraderOffer && this . traderOfferItemQuestLocked ( offer , traderAssorts ) )
{
continue ;
}
const key = offer . items [ 0 ] . _tpl ;
if ( ! offersMap . has ( key ) )
{
offersMap . set ( key , [ ] ) ;
}
offersMap . get ( key ) . push ( offer ) ;
}
}
// get best offer for each item to show on screen
for ( let possibleOffers of offersMap . values ( ) )
{
// Remove offers with locked = true (quest locked) when > 1 possible offers
// single trader item = shows greyed out
// multiple offers for item = is greyed out
if ( possibleOffers . length > 1 )
{
const lockedOffers = this . getLoyaltyLockedOffers ( possibleOffers , pmcProfile ) ;
// Exclude locked offers + above loyalty locked offers if at least 1 was found
const availableOffers = possibleOffers . filter ( x = > ! ( x . locked || lockedOffers . includes ( x . _id ) ) ) ;
if ( availableOffers . length > 0 )
{
possibleOffers = availableOffers ;
}
}
const offer = this . ragfairSortHelper . sortOffers ( possibleOffers , RagfairSort . PRICE , 0 ) [ 0 ] ;
offers . push ( offer ) ;
}
return offers ;
}
/ * *
* Check if offer item is quest locked for current player by looking at sptQuestLocked property in traders barter_scheme
* @param offer Offer to check is quest locked
* @param traderAssorts all trader assorts for player
* @returns true if quest locked
* /
public traderOfferItemQuestLocked ( offer : IRagfairOffer , traderAssorts : Record < string , ITraderAssort > ) : boolean
{
return offer . items ? . some ( i = > traderAssorts [ offer . user . id ] . barter_scheme [ i . _id ] ? . some ( bs1 = > bs1 ? . some ( bs2 = > bs2 . sptQuestLocked ) ) ) ;
}
/ * *
* Has a traders offer ran out of stock to sell to player
* @param offer Offer to check stock of
* @returns true if out of stock
* /
protected traderOutOfStock ( offer : IRagfairOffer ) : boolean
{
if ( offer ? . items ? . length === 0 )
{
return true ;
}
return offer . items [ 0 ] ? . upd ? . StackObjectsCount === 0 ;
}
/ * *
* Check if trader offers ' BuyRestrictionMax value has been reached
* @param offer offer to check restriction properties of
* @returns true if restriction reached , false if no restrictions / not reached
* /
protected traderBuyRestrictionReached ( offer : IRagfairOffer ) : boolean
{
const traderAssorts = this . traderHelper . getTraderAssortsById ( offer . user . id ) . items ;
const assortData = traderAssorts . find ( x = > x . _id === offer . _id ) ;
// No trader assort data
if ( ! assortData )
{
this . logger . warning ( ` Unable to find trader ${ offer . user . nickname } assort for item ${ this . itemHelper . getItemName ( offer . items [ 0 ] . _tpl ) } ${ offer . items [ 0 ] . _tpl } ` ) ;
return false ;
}
// No restriction values
// Can't use !assortData.upd.BuyRestrictionX as value could be 0
if ( assortData . upd . BuyRestrictionMax === undefined || assortData . upd . BuyRestrictionCurrent === undefined )
{
return false ;
}
// Current equals max, limit reached
if ( assortData ? . upd . BuyRestrictionCurrent === assortData . upd . BuyRestrictionMax )
{
return true ;
}
return false ;
}
/ * *
* Get an array of flea offers that are inaccessible to player due to their inadequate loyalty level
* @param offers Offers to check
* @param pmcProfile Players profile with trader loyalty levels
2023-05-29 16:32:06 +01:00
* @returns array of offer ids player cannot see
2023-03-03 15:23:46 +00:00
* /
protected getLoyaltyLockedOffers ( offers : IRagfairOffer [ ] , pmcProfile : IPmcData ) : string [ ]
{
const loyaltyLockedOffers : string [ ] = [ ] ;
for ( const offer of offers )
{
if ( offer . user . memberType === MemberCategory . TRADER )
{
const traderDetails = pmcProfile . TradersInfo [ offer . user . id ] ;
if ( traderDetails . loyaltyLevel < offer . loyaltyLevel )
{
loyaltyLockedOffers . push ( offer . _id ) ;
}
}
}
return loyaltyLockedOffers ;
}
2023-05-29 16:32:06 +01:00
/ * *
* Process all player - listed flea offers for a desired profile
* @param sessionID Session id to process offers for
* @returns true = complete
* /
2023-03-03 15:23:46 +00:00
public processOffersOnProfile ( sessionID : string ) : boolean
{
const timestamp = this . timeUtil . getTimestamp ( ) ;
const profileOffers = this . getProfileOffers ( sessionID ) ;
2023-05-29 16:05:19 +01:00
// No offers, don't do anything
2023-03-03 15:23:46 +00:00
if ( ! profileOffers ? . length )
{
return true ;
}
for ( const offer of profileOffers . values ( ) )
{
if ( offer . sellResult && offer . sellResult . length > 0 && timestamp >= offer . sellResult [ 0 ] . sellTime )
{
// Item sold
let totalItemsCount = 1 ;
let boughtAmount = 1 ;
if ( ! offer . sellInOnePiece )
{
totalItemsCount = offer . items . reduce ( ( sum : number , item ) = > sum += item . upd . StackObjectsCount , 0 ) ;
boughtAmount = offer . sellResult [ 0 ] . amount ;
}
2023-05-29 16:12:27 +01:00
this . increaseProfileRagfairRating ( this . saveServer . getProfile ( sessionID ) , offer . summaryCost / totalItemsCount * boughtAmount ) ;
2023-03-03 15:23:46 +00:00
this . completeOffer ( sessionID , offer , boughtAmount ) ;
offer . sellResult . splice ( 0 , 1 ) ;
2023-05-29 16:05:19 +01:00
// TODO: Send a mail to player informing them offer was sold, text comes from locale with id: "5b55a1f786f77469803bca61 0" (Your offer was sold {buyerNickname})
2023-03-03 15:23:46 +00:00
}
}
return true ;
}
2023-05-29 16:12:27 +01:00
/ * *
* Add amount to players ragfair rating
* @param sessionId Profile to update
* @param amountToIncrementBy Raw amount to add to players ragfair rating ( excluding the reputation gain multiplier )
* /
public increaseProfileRagfairRating ( profile : IAkiProfile , amountToIncrementBy : number ) : void
{
profile . characters . pmc . RagfairInfo . isRatingGrowing = true ;
2023-06-13 19:33:42 +01:00
if ( isNaN ( amountToIncrementBy ) )
{
this . logger . warning ( ` Unable to increment ragfair rating, value was not a number: ${ amountToIncrementBy } ` ) ;
return ;
}
profile . characters . pmc . RagfairInfo . rating += this . ragfairConfig . sell . reputation . gain * amountToIncrementBy ;
2023-05-29 16:12:27 +01:00
}
2023-05-29 16:32:06 +01:00
/ * *
* Return all offers a player has listed on a desired profile
* @param sessionID Session id
* @returns Array of ragfair offers
* /
2023-03-03 15:23:46 +00:00
protected getProfileOffers ( sessionID : string ) : IRagfairOffer [ ]
{
const profile = this . profileHelper . getPmcProfile ( sessionID ) ;
if ( profile . RagfairInfo === undefined || profile . RagfairInfo . offers === undefined )
{
return [ ] ;
}
return profile . RagfairInfo . offers ;
}
2023-05-29 16:32:06 +01:00
/ * *
* Delete an offer from a desired profile
* @param sessionID Session id of profile to delete offer from
* @param offerId Offer id to delete
* /
2023-03-03 15:23:46 +00:00
protected deleteOfferByOfferId ( sessionID : string , offerId : string ) : void
{
const profileRagfairInfo = this . saveServer . getProfile ( sessionID ) . characters . pmc . RagfairInfo ;
const index = profileRagfairInfo . offers . findIndex ( o = > o . _id === offerId ) ;
profileRagfairInfo . offers . splice ( index , 1 ) ;
this . ragfairOfferService . removeOfferById ( offerId ) ;
}
2023-05-29 16:20:41 +01:00
/ * *
* Complete the selling of players ' offer
* @param sessionID Session id
* @param offer Sold offer details
* @param boughtAmount Amount item was purchased for
* @returns Client response
* /
2023-03-03 15:23:46 +00:00
protected completeOffer ( sessionID : string , offer : IRagfairOffer , boughtAmount : number ) : IItemEventRouterResponse
{
const itemTpl = offer . items [ 0 ] . _tpl ;
let itemsToSend = [ ] ;
if ( offer . sellInOnePiece || boughtAmount === offer . items [ 0 ] . upd . StackObjectsCount )
{
this . deleteOfferByOfferId ( sessionID , offer . _id ) ;
}
else
{
offer . items [ 0 ] . upd . StackObjectsCount -= boughtAmount ;
const rootItems = offer . items . filter ( i = > i . parentId === "hideout" ) ;
rootItems . splice ( 0 , 1 ) ;
let removeCount = boughtAmount ;
let idsToRemove : string [ ] = [ ] ;
while ( removeCount > 0 && rootItems . length > 0 )
{
const lastItem = rootItems [ rootItems . length - 1 ] ;
if ( lastItem . upd . StackObjectsCount > removeCount )
{
lastItem . upd . StackObjectsCount -= removeCount ;
removeCount = 0 ;
}
else
{
removeCount -= lastItem . upd . StackObjectsCount ;
idsToRemove . push ( lastItem . _id ) ;
rootItems . splice ( rootItems . length - 1 , 1 ) ;
}
}
let foundNewItems = true ;
while ( foundNewItems )
{
foundNewItems = false ;
// eslint-disable-next-line @typescript-eslint/no-unused-vars
for ( const id of idsToRemove )
{
const newIds = offer . items . filter ( i = > ! idsToRemove . includes ( i . _id ) && idsToRemove . includes ( i . parentId ) ) . map ( i = > i . _id ) ;
if ( newIds . length > 0 )
{
foundNewItems = true ;
idsToRemove = [ . . . idsToRemove , . . . newIds ] ;
}
}
}
if ( idsToRemove . length > 0 )
{
offer . items = offer . items . filter ( i = > ! idsToRemove . includes ( i . _id ) ) ;
}
}
2023-05-29 16:20:41 +01:00
// Assemble the payment item(s)
2023-03-03 15:23:46 +00:00
for ( const requirement of offer . requirements )
{
// Create an item template item
const requestedItem : Item = {
_id : this.hashUtil.generate ( ) ,
_tpl : requirement._tpl ,
upd : { StackObjectsCount : requirement.count * boughtAmount }
} ;
const stacks = this . itemHelper . splitStack ( requestedItem ) ;
for ( const item of stacks )
{
const outItems = [ item ] ;
if ( requirement . onlyFunctional )
{
const presetItems = this . ragfairServerHelper . getPresetItemsByTpl ( item ) ;
if ( presetItems . length )
{
outItems . push ( presetItems [ 0 ] ) ;
}
}
itemsToSend = [ . . . itemsToSend , . . . outItems ] ;
}
}
// Generate a message to inform that item was sold
const globalLocales = this . localeService . getLocaleDb ( ) ;
2023-05-29 16:20:41 +01:00
const soldMessageLocaleGuid = globalLocales [ RagfairOfferHelper . goodSoldTemplate ] ;
if ( ! soldMessageLocaleGuid )
{
2023-07-19 11:00:34 +01:00
this . logger . error ( this . localisationService . getText ( "ragfair-unable_to_find_locale_by_key" , RagfairOfferHelper . goodSoldTemplate ) ) ;
2023-05-29 16:20:41 +01:00
}
// Used to replace tokens in sold message sent to player
2023-03-03 15:23:46 +00:00
const tplVars : ISystemData = {
soldItem : globalLocales [ ` ${ itemTpl } Name ` ] || itemTpl ,
buyerNickname : this.ragfairServerHelper.getNickname ( this . hashUtil . generate ( ) ) ,
itemCount : boughtAmount
} ;
2023-05-29 16:20:41 +01:00
const messageText = soldMessageLocaleGuid . replace ( /{\w+}/g , ( matched ) = >
2023-03-03 15:23:46 +00:00
{
return tplVars [ matched . replace ( /{|}/g , "" ) ] ;
} ) ;
const messageContent = this . dialogueHelper . createMessageContext ( undefined , MessageType . FLEAMARKET_MESSAGE , this . questConfig . redeemTime ) ;
messageContent . text = messageText . replace ( /"/g , "" ) ;
messageContent . ragfair = {
offerId : offer._id ,
count : boughtAmount ,
handbookId : itemTpl
} ;
this . dialogueHelper . addDialogueMessage ( Traders . RAGMAN , messageContent , sessionID , itemsToSend ) ;
2023-05-29 16:20:41 +01:00
2023-03-03 15:23:46 +00:00
return this . eventOutputHolder . getOutput ( sessionID ) ;
}
2023-05-29 16:32:06 +01:00
/ * *
* Should a ragfair offer be visible to the player
* @param info Search request
* @param itemsToAdd ?
* @param traderAssorts Trader assort items
* @param offer The flea offer
* @param pmcProfile Player profile
* @returns True = should be shown to player
* /
2023-03-03 15:23:46 +00:00
public isDisplayableOffer ( info : ISearchRequestData , itemsToAdd : string [ ] , traderAssorts : Record < string , ITraderAssort > , offer : IRagfairOffer , pmcProfile : IPmcData ) : boolean
{
const item = offer . items [ 0 ] ;
const money = offer . requirements [ 0 ] . _tpl ;
const isTraderOffer = offer . user . memberType === MemberCategory . TRADER ;
const isDefaultUserOffer = offer . user . memberType === MemberCategory . DEFAULT ;
if ( pmcProfile . Info . Level < this . databaseServer . getTables ( ) . globals . config . RagFair . minUserLevel && isDefaultUserOffer )
{
// Skip item if player is < global unlock level (default is 15) and item is from a dynamically generated source
return false ;
}
if ( ! ! itemsToAdd && ! itemsToAdd . includes ( item . _tpl ) )
{
// skip items we shouldn't include
return false ;
}
if ( info . offerOwnerType === OfferOwnerType . TRADEROWNERTYPE && ! isTraderOffer )
{
// don't include player offers
return false ;
}
if ( info . offerOwnerType === OfferOwnerType . PLAYEROWNERTYPE && isTraderOffer )
{
// don't include trader offers
return false ;
}
if ( info . oneHourExpiration && offer . endTime - this . timeUtil . getTimestamp ( ) > TimeUtil . oneHourAsSeconds )
{
// offer doesnt expire within an hour
return false ;
}
if ( info . quantityFrom > 0 && info . quantityFrom >= item . upd . StackObjectsCount )
{
// too little items to offer
return false ;
}
if ( info . quantityTo > 0 && info . quantityTo <= item . upd . StackObjectsCount )
{
// too many items to offer
return false ;
}
if ( info . onlyFunctional && this . presetHelper . hasPreset ( item . _tpl ) && offer . items . length === 1 )
{
// don't include non-functional items
return false ;
}
if ( info . buildCount && this . presetHelper . hasPreset ( item . _tpl ) && offer . items . length > 1 )
{
// don't include preset items
return false ;
}
if ( item . upd . MedKit || item . upd . Repairable )
{
const itemQualityPercentage = 100 * this . itemHelper . getItemQualityModifier ( item ) ;
if ( info . conditionFrom > 0 && info . conditionFrom > itemQualityPercentage )
{
// item condition is too low
return false ;
}
if ( info . conditionTo < 100 && info . conditionTo <= itemQualityPercentage )
{
// item condition is too high
return false ;
}
}
// commented out as required search "which is for checking offers that are barters"
// has info.removeBartering as true, this if statement removed barter items.
if ( info . removeBartering && ! this . paymentHelper . isMoneyTpl ( money ) )
{
// don't include barter offers
return false ;
}
if ( info . currency > 0 && this . paymentHelper . isMoneyTpl ( money ) )
{
const currencies = [ "all" , "RUB" , "USD" , "EUR" ] ;
if ( this . ragfairHelper . getCurrencyTag ( money ) !== currencies [ info . currency ] )
{
// don't include item paid in wrong currency
return false ;
}
}
if ( info . priceFrom > 0 && info . priceFrom >= offer . requirementsCost )
{
// price is too low
return false ;
}
if ( info . priceTo > 0 && info . priceTo <= offer . requirementsCost )
{
// price is too high
return false ;
}
if ( isNaN ( offer . requirementsCost ) )
{
// don't include offers with null or NaN in it
return false ;
}
// handle trader items to remove items that are not available to the user right now
// required search for "lamp" shows 4 items, 3 of which are not available to a new player
// filter those out
if ( offer . user . id in this . databaseServer . getTables ( ) . traders )
{
if ( ! ( offer . user . id in traderAssorts ) )
{
// trader not visible on flea market
return false ;
}
if ( ! traderAssorts [ offer . user . id ] . items . find ( ( item ) = >
{
return item . _id === offer . root ;
} ) )
{
// skip (quest) locked items
return false ;
}
}
return true ;
}
}