2021-12-05 17:54:36 +00:00
# include "ChatPacketHandler.h"
# include "PlayerContainer.h"
# include "Database.h"
# include <vector>
# include "PacketUtils.h"
2023-09-21 01:06:28 +00:00
# include "BitStreamUtils.h"
2021-12-05 17:54:36 +00:00
# include "Game.h"
# include "dServer.h"
# include "GeneralUtils.h"
2023-10-21 23:31:55 +00:00
# include "Logger.h"
2023-01-22 23:38:47 +00:00
# include "eAddFriendResponseCode.h"
# include "eAddFriendResponseType.h"
2022-07-13 03:36:06 +00:00
# include "RakString.h"
# include "dConfig.h"
2023-04-25 18:17:40 +00:00
# include "eObjectBits.h"
2023-05-03 21:38:32 +00:00
# include "eConnectionType.h"
# include "eChatMessageType.h"
# include "eChatInternalMessageType.h"
# include "eClientMessageType.h"
# include "eGameMessageType.h"
2021-12-05 17:54:36 +00:00
extern PlayerContainer playerContainer ;
void ChatPacketHandler : : HandleFriendlistRequest ( Packet * packet ) {
//Get from the packet which player we want to do something with:
2023-05-08 11:31:10 +00:00
CINSTREAM_SKIP_HEADER ;
2021-12-05 17:54:36 +00:00
LWOOBJID playerID = 0 ;
inStream . Read ( playerID ) ;
auto player = playerContainer . GetPlayerData ( playerID ) ;
if ( ! player ) return ;
2022-07-11 00:59:07 +00:00
//Get our friends list from the Db. Using a derived table since the friend of a player can be in either column.
2022-07-13 03:36:06 +00:00
std : : unique_ptr < sql : : PreparedStatement > stmt ( Database : : CreatePreppedStmt (
2022-07-11 00:59:07 +00:00
" SELECT fr.requested_player, best_friend, ci.name FROM "
" (SELECT CASE "
2022-07-28 13:39:57 +00:00
" WHEN player_id = ? THEN friend_id "
" WHEN friend_id = ? THEN player_id "
2022-07-11 00:59:07 +00:00
" END AS requested_player, best_friend FROM friends) AS fr "
" JOIN charinfo AS ci ON ci.id = fr.requested_player "
2022-10-22 00:35:12 +00:00
" WHERE fr.requested_player IS NOT NULL AND fr.requested_player != ?; " ) ) ;
2022-07-11 00:59:07 +00:00
stmt - > setUInt ( 1 , static_cast < uint32_t > ( playerID ) ) ;
stmt - > setUInt ( 2 , static_cast < uint32_t > ( playerID ) ) ;
2022-10-22 00:35:12 +00:00
stmt - > setUInt ( 3 , static_cast < uint32_t > ( playerID ) ) ;
2021-12-05 17:54:36 +00:00
std : : vector < FriendData > friends ;
2022-07-13 03:36:06 +00:00
std : : unique_ptr < sql : : ResultSet > res ( stmt - > executeQuery ( ) ) ;
2021-12-05 17:54:36 +00:00
while ( res - > next ( ) ) {
FriendData fd ;
fd . isFTP = false ; // not a thing in DLU
2022-07-11 00:59:07 +00:00
fd . friendID = res - > getUInt ( 1 ) ;
2023-04-25 18:17:40 +00:00
GeneralUtils : : SetBit ( fd . friendID , eObjectBits : : PERSISTENT ) ;
GeneralUtils : : SetBit ( fd . friendID , eObjectBits : : CHARACTER ) ;
2021-12-05 17:54:36 +00:00
2022-07-13 03:36:06 +00:00
fd . isBestFriend = res - > getInt ( 2 ) = = 3 ; //0 = friends, 1 = left_requested, 2 = right_requested, 3 = both_accepted - are now bffs
2022-07-28 13:39:57 +00:00
if ( fd . isBestFriend ) player - > countOfBestFriends + = 1 ;
2022-07-11 00:59:07 +00:00
fd . friendName = res - > getString ( 3 ) ;
2021-12-05 17:54:36 +00:00
//Now check if they're online:
auto fr = playerContainer . GetPlayerData ( fd . friendID ) ;
2022-07-13 03:36:06 +00:00
2021-12-05 17:54:36 +00:00
if ( fr ) {
fd . isOnline = true ;
fd . zoneID = fr - > zoneID ;
//Since this friend is online, we need to update them on the fact that we've just logged in:
2022-07-13 03:36:06 +00:00
SendFriendUpdate ( fr , player , 1 , fd . isBestFriend ) ;
2022-07-28 13:39:57 +00:00
} else {
2021-12-05 17:54:36 +00:00
fd . isOnline = false ;
fd . zoneID = LWOZONEID ( ) ;
}
friends . push_back ( fd ) ;
}
//Now, we need to send the friendlist to the server they came from:
CBITSTREAM ;
2023-09-21 01:06:28 +00:00
BitStreamUtils : : WriteHeader ( bitStream , eConnectionType : : CHAT_INTERNAL , eChatInternalMessageType : : ROUTE_TO_PLAYER ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( playerID ) ;
//portion that will get routed:
2023-09-21 01:06:28 +00:00
BitStreamUtils : : WriteHeader ( bitStream , eConnectionType : : CLIENT , eClientMessageType : : GET_FRIENDS_LIST_RESPONSE ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write < uint8_t > ( 0 ) ;
bitStream . Write < uint16_t > ( 1 ) ; //Length of packet -- just writing one as it doesn't matter, client skips it.
bitStream . Write ( ( uint16_t ) friends . size ( ) ) ;
2022-07-25 02:26:51 +00:00
2021-12-05 17:54:36 +00:00
for ( auto & data : friends ) {
data . Serialize ( bitStream ) ;
}
player - > friends = friends ;
SystemAddress sysAddr = player - > sysAddr ;
SEND_PACKET ;
}
void ChatPacketHandler : : HandleFriendRequest ( Packet * packet ) {
2022-07-13 03:36:06 +00:00
auto maxNumberOfBestFriendsAsString = Game : : config - > GetValue ( " max_number_of_best_friends " ) ;
// If this config option doesn't exist, default to 5 which is what live used.
auto maxNumberOfBestFriends = maxNumberOfBestFriendsAsString ! = " " ? std : : stoi ( maxNumberOfBestFriendsAsString ) : 5U ;
2023-05-08 11:31:10 +00:00
CINSTREAM_SKIP_HEADER ;
2022-07-13 03:36:06 +00:00
LWOOBJID requestorPlayerID ;
inStream . Read ( requestorPlayerID ) ;
uint32_t spacing { } ;
inStream . Read ( spacing ) ;
std : : string playerName = " " ;
uint16_t character ;
bool noMoreLettersInName = false ;
for ( uint32_t j = 0 ; j < 33 ; j + + ) {
inStream . Read ( character ) ;
if ( character = = ' \0 ' ) noMoreLettersInName = true ;
if ( ! noMoreLettersInName ) playerName . push_back ( static_cast < char > ( character ) ) ;
}
2021-12-05 17:54:36 +00:00
2022-07-13 03:36:06 +00:00
char isBestFriendRequest { } ;
inStream . Read ( isBestFriendRequest ) ;
auto requestor = playerContainer . GetPlayerData ( requestorPlayerID ) ;
2023-10-24 09:26:55 +00:00
if ( ! requestor ) {
LOG ( " No requestor player %llu sent to %s found. " , requestorPlayerID , playerName . c_str ( ) ) ;
return ;
}
2022-10-22 00:35:12 +00:00
if ( requestor - > playerName = = playerName ) {
2023-01-22 23:38:47 +00:00
SendFriendResponse ( requestor , requestor , eAddFriendResponseType : : MYTHRAN ) ;
2022-10-22 00:35:12 +00:00
return ;
} ;
2022-07-13 03:36:06 +00:00
std : : unique_ptr < PlayerData > requestee ( playerContainer . GetPlayerData ( playerName ) ) ;
// Check if player is online first
if ( isBestFriendRequest & & ! requestee ) {
for ( auto friendDataCandidate : requestor - > friends ) {
if ( friendDataCandidate . friendName = = playerName ) {
requestee . reset ( new PlayerData ( ) ) ;
// Setup the needed info since you can add a best friend offline.
requestee - > playerID = friendDataCandidate . friendID ;
2022-07-17 03:40:46 +00:00
requestee - > playerName = friendDataCandidate . friendName ;
2022-07-13 03:36:06 +00:00
requestee - > zoneID = LWOZONEID ( ) ;
FriendData requesteeFriendData { } ;
requesteeFriendData . friendID = requestor - > playerID ;
requesteeFriendData . friendName = requestor - > playerName ;
requesteeFriendData . isFTP = false ;
requesteeFriendData . isOnline = false ;
requesteeFriendData . zoneID = requestor - > zoneID ;
requestee - > friends . push_back ( requesteeFriendData ) ;
requestee - > sysAddr = UNASSIGNED_SYSTEM_ADDRESS ;
break ;
}
}
2021-12-05 17:54:36 +00:00
}
2022-07-13 03:36:06 +00:00
2022-07-25 02:26:51 +00:00
// If at this point we dont have a target, then they arent online and we cant send the request.
2022-07-13 03:36:06 +00:00
// Send the response code that corresponds to what the error is.
if ( ! requestee ) {
std : : unique_ptr < sql : : PreparedStatement > nameQuery ( Database : : CreatePreppedStmt ( " SELECT name from charinfo where name = ?; " ) ) ;
nameQuery - > setString ( 1 , playerName ) ;
std : : unique_ptr < sql : : ResultSet > result ( nameQuery - > executeQuery ( ) ) ;
requestee . reset ( new PlayerData ( ) ) ;
2022-07-17 03:40:46 +00:00
requestee - > playerName = playerName ;
2022-07-13 03:36:06 +00:00
2023-01-22 23:38:47 +00:00
SendFriendResponse ( requestor , requestee . get ( ) , result - > next ( ) ? eAddFriendResponseType : : NOTONLINE : eAddFriendResponseType : : INVALIDCHARACTER ) ;
2022-07-13 03:36:06 +00:00
return ;
}
if ( isBestFriendRequest ) {
std : : unique_ptr < sql : : PreparedStatement > friendUpdate ( Database : : CreatePreppedStmt ( " SELECT * FROM friends WHERE (player_id = ? AND friend_id = ?) OR (player_id = ? AND friend_id = ?) LIMIT 1; " ) ) ;
friendUpdate - > setUInt ( 1 , static_cast < uint32_t > ( requestorPlayerID ) ) ;
friendUpdate - > setUInt ( 2 , static_cast < uint32_t > ( requestee - > playerID ) ) ;
friendUpdate - > setUInt ( 3 , static_cast < uint32_t > ( requestee - > playerID ) ) ;
friendUpdate - > setUInt ( 4 , static_cast < uint32_t > ( requestorPlayerID ) ) ;
std : : unique_ptr < sql : : ResultSet > result ( friendUpdate - > executeQuery ( ) ) ;
LWOOBJID queryPlayerID = LWOOBJID_EMPTY ;
LWOOBJID queryFriendID = LWOOBJID_EMPTY ;
uint8_t oldBestFriendStatus { } ;
uint8_t bestFriendStatus { } ;
if ( result - > next ( ) ) {
// Get the IDs
queryPlayerID = result - > getInt ( 1 ) ;
queryFriendID = result - > getInt ( 2 ) ;
oldBestFriendStatus = result - > getInt ( 3 ) ;
bestFriendStatus = oldBestFriendStatus ;
// Set the bits
2023-04-25 18:17:40 +00:00
GeneralUtils : : SetBit ( queryPlayerID , eObjectBits : : CHARACTER ) ;
GeneralUtils : : SetBit ( queryPlayerID , eObjectBits : : PERSISTENT ) ;
GeneralUtils : : SetBit ( queryFriendID , eObjectBits : : CHARACTER ) ;
GeneralUtils : : SetBit ( queryFriendID , eObjectBits : : PERSISTENT ) ;
2022-07-13 03:36:06 +00:00
// Since this player can either be the friend of someone else or be friends with someone else
// their column in the database determines what bit gets set. When the value hits 3, they
// are now best friends with the other player.
if ( queryPlayerID = = requestorPlayerID ) {
bestFriendStatus | = 1ULL < < 0 ;
} else {
bestFriendStatus | = 1ULL < < 1 ;
}
}
// Only do updates if there was a change in the bff status.
if ( oldBestFriendStatus ! = bestFriendStatus ) {
if ( requestee - > countOfBestFriends > = maxNumberOfBestFriends | | requestor - > countOfBestFriends > = maxNumberOfBestFriends ) {
if ( requestee - > countOfBestFriends > = maxNumberOfBestFriends ) {
2023-01-22 23:38:47 +00:00
SendFriendResponse ( requestor , requestee . get ( ) , eAddFriendResponseType : : THEIRFRIENDLISTFULL , false ) ;
2022-07-13 03:36:06 +00:00
}
if ( requestor - > countOfBestFriends > = maxNumberOfBestFriends ) {
2023-01-22 23:38:47 +00:00
SendFriendResponse ( requestor , requestee . get ( ) , eAddFriendResponseType : : YOURFRIENDSLISTFULL , false ) ;
2022-07-13 03:36:06 +00:00
}
} else {
// Then update the database with this new info.
std : : unique_ptr < sql : : PreparedStatement > updateQuery ( Database : : CreatePreppedStmt ( " UPDATE friends SET best_friend = ? WHERE (player_id = ? AND friend_id = ?) OR (player_id = ? AND friend_id = ?) LIMIT 1; " ) ) ;
updateQuery - > setUInt ( 1 , bestFriendStatus ) ;
updateQuery - > setUInt ( 2 , static_cast < uint32_t > ( requestorPlayerID ) ) ;
updateQuery - > setUInt ( 3 , static_cast < uint32_t > ( requestee - > playerID ) ) ;
updateQuery - > setUInt ( 4 , static_cast < uint32_t > ( requestee - > playerID ) ) ;
updateQuery - > setUInt ( 5 , static_cast < uint32_t > ( requestorPlayerID ) ) ;
updateQuery - > executeUpdate ( ) ;
// Sent the best friend update here if the value is 3
if ( bestFriendStatus = = 3U ) {
2022-07-28 13:39:57 +00:00
requestee - > countOfBestFriends + = 1 ;
requestor - > countOfBestFriends + = 1 ;
2023-01-22 23:38:47 +00:00
if ( requestee - > sysAddr ! = UNASSIGNED_SYSTEM_ADDRESS ) SendFriendResponse ( requestee . get ( ) , requestor , eAddFriendResponseType : : ACCEPTED , false , true ) ;
if ( requestor - > sysAddr ! = UNASSIGNED_SYSTEM_ADDRESS ) SendFriendResponse ( requestor , requestee . get ( ) , eAddFriendResponseType : : ACCEPTED , false , true ) ;
2022-07-13 03:36:06 +00:00
for ( auto & friendData : requestor - > friends ) {
if ( friendData . friendID = = requestee - > playerID ) {
friendData . isBestFriend = true ;
}
}
for ( auto & friendData : requestee - > friends ) {
if ( friendData . friendID = = requestor - > playerID ) {
friendData . isBestFriend = true ;
}
}
}
}
} else {
2023-01-22 23:38:47 +00:00
if ( requestor - > sysAddr ! = UNASSIGNED_SYSTEM_ADDRESS ) SendFriendResponse ( requestor , requestee . get ( ) , eAddFriendResponseType : : WAITINGAPPROVAL , true , true ) ;
2022-07-13 03:36:06 +00:00
}
} else {
// Do not send this if we are requesting to be a best friend.
SendFriendRequest ( requestee . get ( ) , requestor ) ;
}
2022-07-25 02:26:51 +00:00
2022-07-13 03:36:06 +00:00
// If the player is actually a player and not a ghost one defined above, release it from being deleted.
if ( requestee - > sysAddr ! = UNASSIGNED_SYSTEM_ADDRESS ) requestee . release ( ) ;
2021-12-05 17:54:36 +00:00
}
void ChatPacketHandler : : HandleFriendResponse ( Packet * packet ) {
2023-05-08 11:31:10 +00:00
CINSTREAM_SKIP_HEADER ;
2021-12-05 17:54:36 +00:00
LWOOBJID playerID ;
inStream . Read ( playerID ) ;
2022-07-13 03:36:06 +00:00
2023-01-22 23:38:47 +00:00
eAddFriendResponseCode clientResponseCode = static_cast < eAddFriendResponseCode > ( packet - > data [ 0x14 ] ) ;
2021-12-05 17:54:36 +00:00
std : : string friendName = PacketUtils : : ReadString ( 0x15 , packet , true ) ;
//Now to try and find both of these:
2022-07-13 03:36:06 +00:00
auto requestor = playerContainer . GetPlayerData ( playerID ) ;
auto requestee = playerContainer . GetPlayerData ( friendName ) ;
if ( ! requestor | | ! requestee ) return ;
2023-01-22 23:38:47 +00:00
eAddFriendResponseType serverResponseCode { } ;
2022-07-13 03:36:06 +00:00
uint8_t isAlreadyBestFriends = 0U ;
// We need to convert this response code to one we can actually send back to the client.
switch ( clientResponseCode ) {
2023-01-22 23:38:47 +00:00
case eAddFriendResponseCode : : ACCEPTED :
serverResponseCode = eAddFriendResponseType : : ACCEPTED ;
2022-07-28 13:39:57 +00:00
break ;
2023-01-22 23:38:47 +00:00
case eAddFriendResponseCode : : BUSY :
serverResponseCode = eAddFriendResponseType : : BUSY ;
2022-07-28 13:39:57 +00:00
break ;
2023-01-22 23:38:47 +00:00
case eAddFriendResponseCode : : CANCELLED :
serverResponseCode = eAddFriendResponseType : : CANCELLED ;
2022-07-28 13:39:57 +00:00
break ;
2023-01-22 23:38:47 +00:00
case eAddFriendResponseCode : : REJECTED :
serverResponseCode = eAddFriendResponseType : : DECLINED ;
2022-07-28 13:39:57 +00:00
break ;
2022-07-11 00:59:07 +00:00
}
2022-07-13 03:36:06 +00:00
// Now that we have handled the base cases, we need to check the other cases.
2023-01-22 23:38:47 +00:00
if ( serverResponseCode = = eAddFriendResponseType : : ACCEPTED ) {
2022-07-13 03:36:06 +00:00
for ( auto friendData : requestor - > friends ) {
if ( friendData . friendID = = requestee - > playerID ) {
2023-01-22 23:38:47 +00:00
serverResponseCode = eAddFriendResponseType : : ALREADYFRIEND ;
2022-07-13 03:36:06 +00:00
if ( friendData . isBestFriend ) {
isAlreadyBestFriends = 1U ;
}
}
}
2022-07-11 00:59:07 +00:00
}
2022-07-13 03:36:06 +00:00
// This message is NOT sent for best friends and is handled differently for those requests.
2023-01-22 23:38:47 +00:00
if ( serverResponseCode = = eAddFriendResponseType : : ACCEPTED ) {
2022-07-13 03:36:06 +00:00
// Add the each player to the others friend list.
FriendData requestorData ;
requestorData . zoneID = requestor - > zoneID ;
requestorData . friendID = requestor - > playerID ;
requestorData . friendName = requestor - > playerName ;
requestorData . isBestFriend = false ;
requestorData . isFTP = false ;
requestorData . isOnline = true ;
requestee - > friends . push_back ( requestorData ) ;
FriendData requesteeData ;
requesteeData . zoneID = requestee - > zoneID ;
requesteeData . friendID = requestee - > playerID ;
requesteeData . friendName = requestee - > playerName ;
requesteeData . isBestFriend = false ;
requesteeData . isFTP = false ;
requesteeData . isOnline = true ;
requestor - > friends . push_back ( requesteeData ) ;
2022-07-25 02:26:51 +00:00
2022-07-13 03:36:06 +00:00
std : : unique_ptr < sql : : PreparedStatement > statement ( Database : : CreatePreppedStmt ( " INSERT IGNORE INTO `friends` (`player_id`, `friend_id`, `best_friend`) VALUES (?,?,?); " ) ) ;
statement - > setUInt ( 1 , static_cast < uint32_t > ( requestor - > playerID ) ) ;
statement - > setUInt ( 2 , static_cast < uint32_t > ( requestee - > playerID ) ) ;
statement - > setInt ( 3 , 0 ) ;
statement - > execute ( ) ;
}
2021-12-05 17:54:36 +00:00
2023-01-22 23:38:47 +00:00
if ( serverResponseCode ! = eAddFriendResponseType : : DECLINED ) SendFriendResponse ( requestor , requestee , serverResponseCode , isAlreadyBestFriends ) ;
if ( serverResponseCode ! = eAddFriendResponseType : : ALREADYFRIEND ) SendFriendResponse ( requestee , requestor , serverResponseCode , isAlreadyBestFriends ) ;
2021-12-05 17:54:36 +00:00
}
void ChatPacketHandler : : HandleRemoveFriend ( Packet * packet ) {
2023-05-08 11:31:10 +00:00
CINSTREAM_SKIP_HEADER ;
2021-12-05 17:54:36 +00:00
LWOOBJID playerID ;
inStream . Read ( playerID ) ;
2022-07-11 00:59:07 +00:00
std : : string friendName = PacketUtils : : ReadString ( 0x14 , packet , true ) ;
2021-12-05 17:54:36 +00:00
//we'll have to query the db here to find the user, since you can delete them while they're offline.
//First, we need to find their ID:
2022-07-13 03:36:06 +00:00
std : : unique_ptr < sql : : PreparedStatement > stmt ( Database : : CreatePreppedStmt ( " SELECT id FROM charinfo WHERE name=? LIMIT 1; " ) ) ;
2021-12-05 17:54:36 +00:00
stmt - > setString ( 1 , friendName . c_str ( ) ) ;
LWOOBJID friendID = 0 ;
2022-07-13 03:36:06 +00:00
std : : unique_ptr < sql : : ResultSet > res ( stmt - > executeQuery ( ) ) ;
2021-12-05 17:54:36 +00:00
while ( res - > next ( ) ) {
2022-07-11 00:59:07 +00:00
friendID = res - > getUInt ( 1 ) ;
2021-12-05 17:54:36 +00:00
}
2022-07-11 00:59:07 +00:00
// Convert friendID to LWOOBJID
2023-04-25 18:17:40 +00:00
GeneralUtils : : SetBit ( friendID , eObjectBits : : PERSISTENT ) ;
GeneralUtils : : SetBit ( friendID , eObjectBits : : CHARACTER ) ;
2022-07-11 00:59:07 +00:00
2022-07-13 03:36:06 +00:00
std : : unique_ptr < sql : : PreparedStatement > deletestmt ( Database : : CreatePreppedStmt ( " DELETE FROM friends WHERE (player_id = ? AND friend_id = ?) OR (player_id = ? AND friend_id = ?) LIMIT 1; " ) ) ;
2022-07-11 00:59:07 +00:00
deletestmt - > setUInt ( 1 , static_cast < uint32_t > ( playerID ) ) ;
deletestmt - > setUInt ( 2 , static_cast < uint32_t > ( friendID ) ) ;
deletestmt - > setUInt ( 3 , static_cast < uint32_t > ( friendID ) ) ;
deletestmt - > setUInt ( 4 , static_cast < uint32_t > ( playerID ) ) ;
2021-12-05 17:54:36 +00:00
deletestmt - > execute ( ) ;
//Now, we need to send an update to notify the sender (and possibly, receiver) that their friendship has been ended:
auto goonA = playerContainer . GetPlayerData ( playerID ) ;
if ( goonA ) {
2022-07-11 00:59:07 +00:00
// Remove the friend from our list of friends
for ( auto friendData = goonA - > friends . begin ( ) ; friendData ! = goonA - > friends . end ( ) ; friendData + + ) {
if ( ( * friendData ) . friendID = = friendID ) {
2022-07-13 03:36:06 +00:00
if ( ( * friendData ) . isBestFriend ) - - goonA - > countOfBestFriends ;
2022-07-11 00:59:07 +00:00
goonA - > friends . erase ( friendData ) ;
break ;
}
}
2021-12-05 17:54:36 +00:00
SendRemoveFriend ( goonA , friendName , true ) ;
}
2022-07-11 00:59:07 +00:00
2021-12-05 17:54:36 +00:00
auto goonB = playerContainer . GetPlayerData ( friendID ) ;
if ( ! goonB ) return ;
2022-07-11 00:59:07 +00:00
// Do it again for other person
for ( auto friendData = goonB - > friends . begin ( ) ; friendData ! = goonB - > friends . end ( ) ; friendData + + ) {
if ( ( * friendData ) . friendID = = playerID ) {
2022-07-13 03:36:06 +00:00
if ( ( * friendData ) . isBestFriend ) - - goonB - > countOfBestFriends ;
2022-07-11 00:59:07 +00:00
goonB - > friends . erase ( friendData ) ;
break ;
}
}
2021-12-05 17:54:36 +00:00
std : : string goonAName = GeneralUtils : : UTF16ToWTF8 ( playerContainer . GetName ( playerID ) ) ;
SendRemoveFriend ( goonB , goonAName , true ) ;
}
2022-07-28 13:39:57 +00:00
void ChatPacketHandler : : HandleChatMessage ( Packet * packet ) {
2023-05-08 11:31:10 +00:00
CINSTREAM_SKIP_HEADER ;
2021-12-05 17:54:36 +00:00
LWOOBJID playerID = LWOOBJID_EMPTY ;
inStream . Read ( playerID ) ;
auto * sender = playerContainer . GetPlayerData ( playerID ) ;
if ( sender = = nullptr ) return ;
if ( playerContainer . GetIsMuted ( sender ) ) return ;
2022-07-17 03:40:46 +00:00
const auto senderName = std : : string ( sender - > playerName . c_str ( ) ) ;
2021-12-05 17:54:36 +00:00
inStream . SetReadOffset ( 0x14 * 8 ) ;
uint8_t channel = 0 ;
inStream . Read ( channel ) ;
2022-07-25 02:26:51 +00:00
2022-11-04 19:28:19 +00:00
std : : string message = PacketUtils : : ReadString ( 0x66 , packet , true , 512 ) ;
2021-12-05 17:54:36 +00:00
2023-10-21 23:31:55 +00:00
LOG ( " Got a message from (%s) [%d]: %s " , senderName . c_str ( ) , channel , message . c_str ( ) ) ;
2021-12-05 17:54:36 +00:00
if ( channel ! = 8 ) return ;
auto * team = playerContainer . GetTeam ( playerID ) ;
if ( team = = nullptr ) return ;
2022-07-28 13:39:57 +00:00
for ( const auto memberId : team - > memberIDs ) {
2021-12-05 17:54:36 +00:00
auto * otherMember = playerContainer . GetPlayerData ( memberId ) ;
if ( otherMember = = nullptr ) return ;
2022-07-17 03:40:46 +00:00
const auto otherName = std : : string ( otherMember - > playerName . c_str ( ) ) ;
2021-12-05 17:54:36 +00:00
CBITSTREAM ;
2023-09-21 01:06:28 +00:00
BitStreamUtils : : WriteHeader ( bitStream , eConnectionType : : CHAT_INTERNAL , eChatInternalMessageType : : ROUTE_TO_PLAYER ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( otherMember - > playerID ) ;
2023-09-21 01:06:28 +00:00
BitStreamUtils : : WriteHeader ( bitStream , eConnectionType : : CHAT , eChatMessageType : : PRIVATE_CHAT_MESSAGE ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( otherMember - > playerID ) ;
bitStream . Write < uint8_t > ( 8 ) ;
bitStream . Write < unsigned int > ( 69 ) ;
2023-09-21 01:06:28 +00:00
bitStream . Write ( LUWString ( senderName ) ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( sender - > playerID ) ;
bitStream . Write < uint16_t > ( 0 ) ;
bitStream . Write < uint8_t > ( 0 ) ; //not mythran nametag
2023-09-21 01:06:28 +00:00
bitStream . Write ( LUWString ( otherName ) ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write < uint8_t > ( 0 ) ; //not mythran for receiver
bitStream . Write < uint8_t > ( 0 ) ; //teams?
2023-09-21 01:06:28 +00:00
bitStream . Write ( LUWString ( message , 512 ) ) ;
2021-12-05 17:54:36 +00:00
SystemAddress sysAddr = otherMember - > sysAddr ;
SEND_PACKET ;
}
}
void ChatPacketHandler : : HandlePrivateChatMessage ( Packet * packet ) {
2023-09-21 01:06:28 +00:00
LWOOBJID senderID = PacketUtils : : ReadS64 ( 0x08 , packet ) ;
2021-12-05 17:54:36 +00:00
std : : string receiverName = PacketUtils : : ReadString ( 0x66 , packet , true ) ;
2022-11-04 19:28:19 +00:00
std : : string message = PacketUtils : : ReadString ( 0xAA , packet , true , 512 ) ;
2021-12-05 17:54:36 +00:00
//Get the bois:
auto goonA = playerContainer . GetPlayerData ( senderID ) ;
auto goonB = playerContainer . GetPlayerData ( receiverName ) ;
if ( ! goonA | | ! goonB ) return ;
if ( playerContainer . GetIsMuted ( goonA ) ) return ;
2022-07-17 03:40:46 +00:00
std : : string goonAName = goonA - > playerName . c_str ( ) ;
std : : string goonBName = goonB - > playerName . c_str ( ) ;
2021-12-05 17:54:36 +00:00
//To the sender:
{
CBITSTREAM ;
2023-09-21 01:06:28 +00:00
BitStreamUtils : : WriteHeader ( bitStream , eConnectionType : : CHAT_INTERNAL , eChatInternalMessageType : : ROUTE_TO_PLAYER ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( goonA - > playerID ) ;
2023-09-21 01:06:28 +00:00
BitStreamUtils : : WriteHeader ( bitStream , eConnectionType : : CHAT , eChatMessageType : : PRIVATE_CHAT_MESSAGE ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( goonA - > playerID ) ;
bitStream . Write < uint8_t > ( 7 ) ;
bitStream . Write < unsigned int > ( 69 ) ;
2023-09-21 01:06:28 +00:00
bitStream . Write ( LUWString ( goonAName ) ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( goonA - > playerID ) ;
bitStream . Write < uint16_t > ( 0 ) ;
bitStream . Write < uint8_t > ( 0 ) ; //not mythran nametag
2023-09-21 01:06:28 +00:00
bitStream . Write ( LUWString ( goonBName ) ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write < uint8_t > ( 0 ) ; //not mythran for receiver
bitStream . Write < uint8_t > ( 0 ) ; //success
2023-09-21 01:06:28 +00:00
bitStream . Write ( LUWString ( message , 512 ) ) ;
2021-12-05 17:54:36 +00:00
SystemAddress sysAddr = goonA - > sysAddr ;
SEND_PACKET ;
}
//To the receiver:
{
CBITSTREAM ;
2023-09-21 01:06:28 +00:00
BitStreamUtils : : WriteHeader ( bitStream , eConnectionType : : CHAT_INTERNAL , eChatInternalMessageType : : ROUTE_TO_PLAYER ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( goonB - > playerID ) ;
2023-09-21 01:06:28 +00:00
BitStreamUtils : : WriteHeader ( bitStream , eConnectionType : : CHAT , eChatMessageType : : PRIVATE_CHAT_MESSAGE ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( goonA - > playerID ) ;
bitStream . Write < uint8_t > ( 7 ) ;
bitStream . Write < unsigned int > ( 69 ) ;
2023-09-21 01:06:28 +00:00
bitStream . Write ( LUWString ( goonAName ) ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( goonA - > playerID ) ;
bitStream . Write < uint16_t > ( 0 ) ;
bitStream . Write < uint8_t > ( 0 ) ; //not mythran nametag
2023-09-21 01:06:28 +00:00
bitStream . Write ( LUWString ( goonBName ) ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write < uint8_t > ( 0 ) ; //not mythran for receiver
bitStream . Write < uint8_t > ( 3 ) ; //new whisper
2023-09-21 01:06:28 +00:00
bitStream . Write ( LUWString ( message , 512 ) ) ;
2021-12-05 17:54:36 +00:00
SystemAddress sysAddr = goonB - > sysAddr ;
SEND_PACKET ;
}
}
2022-07-28 13:39:57 +00:00
void ChatPacketHandler : : HandleTeamInvite ( Packet * packet ) {
2023-05-08 11:31:10 +00:00
CINSTREAM_SKIP_HEADER ;
2021-12-05 17:54:36 +00:00
LWOOBJID playerID ;
inStream . Read ( playerID ) ;
std : : string invitedPlayer = PacketUtils : : ReadString ( 0x14 , packet , true ) ;
auto * player = playerContainer . GetPlayerData ( playerID ) ;
2022-07-28 13:39:57 +00:00
if ( player = = nullptr ) {
2021-12-05 17:54:36 +00:00
return ;
}
auto * team = playerContainer . GetTeam ( playerID ) ;
2022-07-28 13:39:57 +00:00
if ( team = = nullptr ) {
2021-12-05 17:54:36 +00:00
team = playerContainer . CreateTeam ( playerID ) ;
}
auto * other = playerContainer . GetPlayerData ( invitedPlayer ) ;
2022-07-28 13:39:57 +00:00
if ( other = = nullptr ) {
2021-12-05 17:54:36 +00:00
return ;
}
2022-07-28 13:39:57 +00:00
if ( playerContainer . GetTeam ( other - > playerID ) ! = nullptr ) {
2021-12-05 17:54:36 +00:00
return ;
}
2021-12-24 00:25:52 +00:00
if ( team - > memberIDs . size ( ) > 3 ) {
// no more teams greater than 4
2023-10-21 23:31:55 +00:00
LOG ( " Someone tried to invite a 5th player to a team " ) ;
2021-12-24 11:57:31 +00:00
return ;
2021-12-24 00:25:52 +00:00
}
2021-12-05 17:54:36 +00:00
SendTeamInvite ( other , player ) ;
2023-10-21 23:31:55 +00:00
LOG ( " Got team invite: %llu -> %s " , playerID , invitedPlayer . c_str ( ) ) ;
2021-12-05 17:54:36 +00:00
}
2022-07-28 13:39:57 +00:00
void ChatPacketHandler : : HandleTeamInviteResponse ( Packet * packet ) {
2023-05-08 11:31:10 +00:00
CINSTREAM_SKIP_HEADER ;
2021-12-05 17:54:36 +00:00
LWOOBJID playerID = LWOOBJID_EMPTY ;
inStream . Read ( playerID ) ;
uint32_t size = 0 ;
inStream . Read ( size ) ;
char declined = 0 ;
inStream . Read ( declined ) ;
LWOOBJID leaderID = LWOOBJID_EMPTY ;
inStream . Read ( leaderID ) ;
2023-10-21 23:31:55 +00:00
LOG ( " Accepted invite: %llu -> %llu (%d) " , playerID , leaderID , declined ) ;
2021-12-05 17:54:36 +00:00
2022-07-28 13:39:57 +00:00
if ( declined ) {
2021-12-05 17:54:36 +00:00
return ;
}
auto * team = playerContainer . GetTeam ( leaderID ) ;
2022-07-28 13:39:57 +00:00
if ( team = = nullptr ) {
2023-10-21 23:31:55 +00:00
LOG ( " Failed to find team for leader (%llu) " , leaderID ) ;
2021-12-05 17:54:36 +00:00
team = playerContainer . GetTeam ( playerID ) ;
}
2022-07-25 02:26:51 +00:00
2022-07-28 13:39:57 +00:00
if ( team = = nullptr ) {
2023-10-21 23:31:55 +00:00
LOG ( " Failed to find team for player (%llu) " , playerID ) ;
2021-12-05 17:54:36 +00:00
return ;
}
playerContainer . AddMember ( team , playerID ) ;
}
2022-07-28 13:39:57 +00:00
void ChatPacketHandler : : HandleTeamLeave ( Packet * packet ) {
2023-05-08 11:31:10 +00:00
CINSTREAM_SKIP_HEADER ;
2021-12-05 17:54:36 +00:00
LWOOBJID playerID = LWOOBJID_EMPTY ;
inStream . Read ( playerID ) ;
uint32_t size = 0 ;
inStream . Read ( size ) ;
auto * team = playerContainer . GetTeam ( playerID ) ;
2023-10-21 23:31:55 +00:00
LOG ( " (%llu) leaving team " , playerID ) ;
2021-12-05 17:54:36 +00:00
2022-07-28 13:39:57 +00:00
if ( team ! = nullptr ) {
2021-12-05 17:54:36 +00:00
playerContainer . RemoveMember ( team , playerID , false , false , true ) ;
}
}
2022-07-28 13:39:57 +00:00
void ChatPacketHandler : : HandleTeamKick ( Packet * packet ) {
2023-05-08 11:31:10 +00:00
CINSTREAM_SKIP_HEADER ;
2021-12-05 17:54:36 +00:00
LWOOBJID playerID = LWOOBJID_EMPTY ;
inStream . Read ( playerID ) ;
2022-07-25 02:26:51 +00:00
2021-12-05 17:54:36 +00:00
std : : string kickedPlayer = PacketUtils : : ReadString ( 0x14 , packet , true ) ;
2023-10-21 23:31:55 +00:00
LOG ( " (%llu) kicking (%s) from team " , playerID , kickedPlayer . c_str ( ) ) ;
2021-12-05 17:54:36 +00:00
auto * kicked = playerContainer . GetPlayerData ( kickedPlayer ) ;
LWOOBJID kickedId = LWOOBJID_EMPTY ;
2022-07-28 13:39:57 +00:00
if ( kicked ! = nullptr ) {
2021-12-05 17:54:36 +00:00
kickedId = kicked - > playerID ;
2022-07-28 13:39:57 +00:00
} else {
2022-08-02 13:56:20 +00:00
kickedId = playerContainer . GetId ( GeneralUtils : : UTF8ToUTF16 ( kickedPlayer ) ) ;
2021-12-05 17:54:36 +00:00
}
if ( kickedId = = LWOOBJID_EMPTY ) return ;
auto * team = playerContainer . GetTeam ( playerID ) ;
2022-07-28 13:39:57 +00:00
if ( team ! = nullptr ) {
2021-12-05 17:54:36 +00:00
if ( team - > leaderID ! = playerID | | team - > leaderID = = kickedId ) return ;
playerContainer . RemoveMember ( team , kickedId , false , true , false ) ;
}
}
2022-07-28 13:39:57 +00:00
void ChatPacketHandler : : HandleTeamPromote ( Packet * packet ) {
2023-05-08 11:31:10 +00:00
CINSTREAM_SKIP_HEADER ;
2021-12-05 17:54:36 +00:00
LWOOBJID playerID = LWOOBJID_EMPTY ;
inStream . Read ( playerID ) ;
2022-07-25 02:26:51 +00:00
2021-12-05 17:54:36 +00:00
std : : string promotedPlayer = PacketUtils : : ReadString ( 0x14 , packet , true ) ;
2023-10-21 23:31:55 +00:00
LOG ( " (%llu) promoting (%s) to team leader " , playerID , promotedPlayer . c_str ( ) ) ;
2021-12-05 17:54:36 +00:00
auto * promoted = playerContainer . GetPlayerData ( promotedPlayer ) ;
if ( promoted = = nullptr ) return ;
auto * team = playerContainer . GetTeam ( playerID ) ;
2022-07-28 13:39:57 +00:00
if ( team ! = nullptr ) {
2021-12-05 17:54:36 +00:00
if ( team - > leaderID ! = playerID ) return ;
playerContainer . PromoteMember ( team , promoted - > playerID ) ;
}
}
2022-07-28 13:39:57 +00:00
void ChatPacketHandler : : HandleTeamLootOption ( Packet * packet ) {
2023-05-08 11:31:10 +00:00
CINSTREAM_SKIP_HEADER ;
2021-12-05 17:54:36 +00:00
LWOOBJID playerID = LWOOBJID_EMPTY ;
inStream . Read ( playerID ) ;
uint32_t size = 0 ;
inStream . Read ( size ) ;
2022-07-25 02:26:51 +00:00
2021-12-05 17:54:36 +00:00
char option ;
inStream . Read ( option ) ;
auto * team = playerContainer . GetTeam ( playerID ) ;
2022-07-28 13:39:57 +00:00
if ( team ! = nullptr ) {
2021-12-05 17:54:36 +00:00
if ( team - > leaderID ! = playerID ) return ;
team - > lootFlag = option ;
playerContainer . TeamStatusUpdate ( team ) ;
2022-07-25 02:26:51 +00:00
2021-12-05 17:54:36 +00:00
playerContainer . UpdateTeamsOnWorld ( team , false ) ;
}
}
2022-07-28 13:39:57 +00:00
void ChatPacketHandler : : HandleTeamStatusRequest ( Packet * packet ) {
2023-05-08 11:31:10 +00:00
CINSTREAM_SKIP_HEADER ;
2021-12-05 17:54:36 +00:00
LWOOBJID playerID = LWOOBJID_EMPTY ;
inStream . Read ( playerID ) ;
auto * team = playerContainer . GetTeam ( playerID ) ;
auto * data = playerContainer . GetPlayerData ( playerID ) ;
2022-07-28 13:39:57 +00:00
if ( team ! = nullptr & & data ! = nullptr ) {
if ( team - > local & & data - > zoneID . GetMapID ( ) ! = team - > zoneId . GetMapID ( ) & & data - > zoneID . GetCloneID ( ) ! = team - > zoneId . GetCloneID ( ) ) {
2021-12-05 17:54:36 +00:00
playerContainer . RemoveMember ( team , playerID , false , false , true , true ) ;
return ;
}
2022-07-28 13:39:57 +00:00
if ( team - > memberIDs . size ( ) < = 1 & & ! team - > local ) {
2021-12-05 17:54:36 +00:00
playerContainer . DisbandTeam ( team ) ;
return ;
}
2022-07-28 13:39:57 +00:00
if ( ! team - > local ) {
2021-12-05 17:54:36 +00:00
ChatPacketHandler : : SendTeamSetLeader ( data , team - > leaderID ) ;
2022-07-28 13:39:57 +00:00
} else {
2021-12-05 17:54:36 +00:00
ChatPacketHandler : : SendTeamSetLeader ( data , LWOOBJID_EMPTY ) ;
}
playerContainer . TeamStatusUpdate ( team ) ;
2022-08-02 13:56:20 +00:00
const auto leaderName = GeneralUtils : : UTF8ToUTF16 ( data - > playerName ) ;
2021-12-05 17:54:36 +00:00
2022-07-28 13:39:57 +00:00
for ( const auto memberId : team - > memberIDs ) {
2021-12-05 17:54:36 +00:00
auto * otherMember = playerContainer . GetPlayerData ( memberId ) ;
if ( memberId = = playerID ) continue ;
const auto memberName = playerContainer . GetName ( memberId ) ;
2022-07-25 02:26:51 +00:00
2022-07-28 13:39:57 +00:00
if ( otherMember ! = nullptr ) {
2021-12-05 17:54:36 +00:00
ChatPacketHandler : : SendTeamSetOffWorldFlag ( otherMember , data - > playerID , data - > zoneID ) ;
}
ChatPacketHandler : : SendTeamAddPlayer ( data , false , team - > local , false , memberId , memberName , otherMember ! = nullptr ? otherMember - > zoneID : LWOZONEID ( 0 , 0 , 0 ) ) ;
}
playerContainer . UpdateTeamsOnWorld ( team , false ) ;
}
}
2022-07-28 13:39:57 +00:00
void ChatPacketHandler : : SendTeamInvite ( PlayerData * receiver , PlayerData * sender ) {
2021-12-05 17:54:36 +00:00
CBITSTREAM ;
2023-09-21 01:06:28 +00:00
BitStreamUtils : : WriteHeader ( bitStream , eConnectionType : : CHAT_INTERNAL , eChatInternalMessageType : : ROUTE_TO_PLAYER ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( receiver - > playerID ) ;
//portion that will get routed:
2023-09-21 01:06:28 +00:00
BitStreamUtils : : WriteHeader ( bitStream , eConnectionType : : CLIENT , eClientMessageType : : TEAM_INVITE ) ;
2021-12-05 17:54:36 +00:00
2023-09-21 01:06:28 +00:00
bitStream . Write ( LUWString ( sender - > playerName . c_str ( ) ) ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( sender - > playerID ) ;
SystemAddress sysAddr = receiver - > sysAddr ;
SEND_PACKET ;
}
2022-07-28 13:39:57 +00:00
void ChatPacketHandler : : SendTeamInviteConfirm ( PlayerData * receiver , bool bLeaderIsFreeTrial , LWOOBJID i64LeaderID , LWOZONEID i64LeaderZoneID , uint8_t ucLootFlag , uint8_t ucNumOfOtherPlayers , uint8_t ucResponseCode , std : : u16string wsLeaderName ) {
2021-12-05 17:54:36 +00:00
CBITSTREAM ;
2023-09-21 01:06:28 +00:00
BitStreamUtils : : WriteHeader ( bitStream , eConnectionType : : CHAT_INTERNAL , eChatInternalMessageType : : ROUTE_TO_PLAYER ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( receiver - > playerID ) ;
//portion that will get routed:
2022-08-05 13:40:12 +00:00
CMSGHEADER ;
2021-12-05 17:54:36 +00:00
2022-08-05 13:40:12 +00:00
bitStream . Write ( receiver - > playerID ) ;
2023-05-03 21:38:32 +00:00
bitStream . Write ( eGameMessageType : : TEAM_INVITE_CONFIRM ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( bLeaderIsFreeTrial ) ;
bitStream . Write ( i64LeaderID ) ;
bitStream . Write ( i64LeaderZoneID ) ;
bitStream . Write < uint32_t > ( 0 ) ; // BinaryBuffe, no clue what's in here
bitStream . Write ( ucLootFlag ) ;
bitStream . Write ( ucNumOfOtherPlayers ) ;
bitStream . Write ( ucResponseCode ) ;
bitStream . Write ( static_cast < uint32_t > ( wsLeaderName . size ( ) ) ) ;
2022-07-28 13:39:57 +00:00
for ( const auto character : wsLeaderName ) {
2021-12-05 17:54:36 +00:00
bitStream . Write ( character ) ;
}
SystemAddress sysAddr = receiver - > sysAddr ;
SEND_PACKET ;
}
2022-07-28 13:39:57 +00:00
void ChatPacketHandler : : SendTeamStatus ( PlayerData * receiver , LWOOBJID i64LeaderID , LWOZONEID i64LeaderZoneID , uint8_t ucLootFlag , uint8_t ucNumOfOtherPlayers , std : : u16string wsLeaderName ) {
2021-12-05 17:54:36 +00:00
CBITSTREAM ;
2023-09-21 01:06:28 +00:00
BitStreamUtils : : WriteHeader ( bitStream , eConnectionType : : CHAT_INTERNAL , eChatInternalMessageType : : ROUTE_TO_PLAYER ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( receiver - > playerID ) ;
//portion that will get routed:
2022-08-05 13:40:12 +00:00
CMSGHEADER ;
2021-12-05 17:54:36 +00:00
2022-08-05 13:40:12 +00:00
bitStream . Write ( receiver - > playerID ) ;
2023-05-03 21:38:32 +00:00
bitStream . Write ( eGameMessageType : : TEAM_GET_STATUS_RESPONSE ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( i64LeaderID ) ;
bitStream . Write ( i64LeaderZoneID ) ;
bitStream . Write < uint32_t > ( 0 ) ; // BinaryBuffe, no clue what's in here
bitStream . Write ( ucLootFlag ) ;
bitStream . Write ( ucNumOfOtherPlayers ) ;
bitStream . Write ( static_cast < uint32_t > ( wsLeaderName . size ( ) ) ) ;
2022-07-28 13:39:57 +00:00
for ( const auto character : wsLeaderName ) {
2021-12-05 17:54:36 +00:00
bitStream . Write ( character ) ;
}
SystemAddress sysAddr = receiver - > sysAddr ;
SEND_PACKET ;
}
2022-07-28 13:39:57 +00:00
void ChatPacketHandler : : SendTeamSetLeader ( PlayerData * receiver , LWOOBJID i64PlayerID ) {
2021-12-05 17:54:36 +00:00
CBITSTREAM ;
2023-09-21 01:06:28 +00:00
BitStreamUtils : : WriteHeader ( bitStream , eConnectionType : : CHAT_INTERNAL , eChatInternalMessageType : : ROUTE_TO_PLAYER ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( receiver - > playerID ) ;
//portion that will get routed:
2022-08-05 13:40:12 +00:00
CMSGHEADER ;
2021-12-05 17:54:36 +00:00
2022-08-05 13:40:12 +00:00
bitStream . Write ( receiver - > playerID ) ;
2023-05-03 21:38:32 +00:00
bitStream . Write ( eGameMessageType : : TEAM_SET_LEADER ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( i64PlayerID ) ;
SystemAddress sysAddr = receiver - > sysAddr ;
SEND_PACKET ;
}
2022-07-28 13:39:57 +00:00
void ChatPacketHandler : : SendTeamAddPlayer ( PlayerData * receiver , bool bIsFreeTrial , bool bLocal , bool bNoLootOnDeath , LWOOBJID i64PlayerID , std : : u16string wsPlayerName , LWOZONEID zoneID ) {
2021-12-05 17:54:36 +00:00
CBITSTREAM ;
2023-09-21 01:06:28 +00:00
BitStreamUtils : : WriteHeader ( bitStream , eConnectionType : : CHAT_INTERNAL , eChatInternalMessageType : : ROUTE_TO_PLAYER ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( receiver - > playerID ) ;
//portion that will get routed:
2022-08-05 13:40:12 +00:00
CMSGHEADER ;
2021-12-05 17:54:36 +00:00
2022-08-05 13:40:12 +00:00
bitStream . Write ( receiver - > playerID ) ;
2023-05-03 21:38:32 +00:00
bitStream . Write ( eGameMessageType : : TEAM_ADD_PLAYER ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( bIsFreeTrial ) ;
bitStream . Write ( bLocal ) ;
bitStream . Write ( bNoLootOnDeath ) ;
bitStream . Write ( i64PlayerID ) ;
bitStream . Write ( static_cast < uint32_t > ( wsPlayerName . size ( ) ) ) ;
2022-07-28 13:39:57 +00:00
for ( const auto character : wsPlayerName ) {
2021-12-05 17:54:36 +00:00
bitStream . Write ( character ) ;
}
bitStream . Write1 ( ) ;
2022-07-28 13:39:57 +00:00
if ( receiver - > zoneID . GetCloneID ( ) = = zoneID . GetCloneID ( ) ) {
2021-12-05 17:54:36 +00:00
zoneID = LWOZONEID ( zoneID . GetMapID ( ) , zoneID . GetInstanceID ( ) , 0 ) ;
}
bitStream . Write ( zoneID ) ;
SystemAddress sysAddr = receiver - > sysAddr ;
SEND_PACKET ;
}
2022-07-28 13:39:57 +00:00
void ChatPacketHandler : : SendTeamRemovePlayer ( PlayerData * receiver , bool bDisband , bool bIsKicked , bool bIsLeaving , bool bLocal , LWOOBJID i64LeaderID , LWOOBJID i64PlayerID , std : : u16string wsPlayerName ) {
2021-12-05 17:54:36 +00:00
CBITSTREAM ;
2023-09-21 01:06:28 +00:00
BitStreamUtils : : WriteHeader ( bitStream , eConnectionType : : CHAT_INTERNAL , eChatInternalMessageType : : ROUTE_TO_PLAYER ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( receiver - > playerID ) ;
//portion that will get routed:
2022-08-05 13:40:12 +00:00
CMSGHEADER ;
2021-12-05 17:54:36 +00:00
2022-08-05 13:40:12 +00:00
bitStream . Write ( receiver - > playerID ) ;
2023-05-03 21:38:32 +00:00
bitStream . Write ( eGameMessageType : : TEAM_REMOVE_PLAYER ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( bDisband ) ;
bitStream . Write ( bIsKicked ) ;
bitStream . Write ( bIsLeaving ) ;
bitStream . Write ( bLocal ) ;
bitStream . Write ( i64LeaderID ) ;
bitStream . Write ( i64PlayerID ) ;
bitStream . Write ( static_cast < uint32_t > ( wsPlayerName . size ( ) ) ) ;
2022-07-28 13:39:57 +00:00
for ( const auto character : wsPlayerName ) {
2021-12-05 17:54:36 +00:00
bitStream . Write ( character ) ;
}
SystemAddress sysAddr = receiver - > sysAddr ;
SEND_PACKET ;
}
2022-07-28 13:39:57 +00:00
void ChatPacketHandler : : SendTeamSetOffWorldFlag ( PlayerData * receiver , LWOOBJID i64PlayerID , LWOZONEID zoneID ) {
2021-12-05 17:54:36 +00:00
CBITSTREAM ;
2023-09-21 01:06:28 +00:00
BitStreamUtils : : WriteHeader ( bitStream , eConnectionType : : CHAT_INTERNAL , eChatInternalMessageType : : ROUTE_TO_PLAYER ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( receiver - > playerID ) ;
//portion that will get routed:
2022-08-05 13:40:12 +00:00
CMSGHEADER ;
2021-12-05 17:54:36 +00:00
2022-08-05 13:40:12 +00:00
bitStream . Write ( receiver - > playerID ) ;
2023-05-03 21:38:32 +00:00
bitStream . Write ( eGameMessageType : : TEAM_SET_OFF_WORLD_FLAG ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( i64PlayerID ) ;
2022-07-28 13:39:57 +00:00
if ( receiver - > zoneID . GetCloneID ( ) = = zoneID . GetCloneID ( ) ) {
2021-12-05 17:54:36 +00:00
zoneID = LWOZONEID ( zoneID . GetMapID ( ) , zoneID . GetInstanceID ( ) , 0 ) ;
}
bitStream . Write ( zoneID ) ;
SystemAddress sysAddr = receiver - > sysAddr ;
SEND_PACKET ;
}
2022-07-13 03:36:06 +00:00
void ChatPacketHandler : : SendFriendUpdate ( PlayerData * friendData , PlayerData * playerData , uint8_t notifyType , uint8_t isBestFriend ) {
2021-12-05 17:54:36 +00:00
/*chat notification is displayed if log in / out and friend is updated in friends list
[ u8 ] - update type
Update types
0 - friend logged out
1 - friend logged in
2 - friend changed world / updated
[ wstring ] - Name of friend
[ u16 ] - World ID
[ u16 ] - World Instance
[ u32 ] - World Clone
[ bool ] - is best friend
[ bool ] - is FTP */
CBITSTREAM ;
2023-09-21 01:06:28 +00:00
BitStreamUtils : : WriteHeader ( bitStream , eConnectionType : : CHAT_INTERNAL , eChatInternalMessageType : : ROUTE_TO_PLAYER ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( friendData - > playerID ) ;
//portion that will get routed:
2023-09-21 01:06:28 +00:00
BitStreamUtils : : WriteHeader ( bitStream , eConnectionType : : CLIENT , eClientMessageType : : UPDATE_FRIEND_NOTIFY ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write < uint8_t > ( notifyType ) ;
2022-07-17 03:40:46 +00:00
std : : string playerName = playerData - > playerName . c_str ( ) ;
2021-12-05 17:54:36 +00:00
2023-09-21 01:06:28 +00:00
bitStream . Write ( LUWString ( playerName ) ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( playerData - > zoneID . GetMapID ( ) ) ;
bitStream . Write ( playerData - > zoneID . GetInstanceID ( ) ) ;
2022-07-28 13:39:57 +00:00
if ( playerData - > zoneID . GetCloneID ( ) = = friendData - > zoneID . GetCloneID ( ) ) {
2021-12-05 17:54:36 +00:00
bitStream . Write ( 0 ) ;
2022-07-28 13:39:57 +00:00
} else {
2021-12-05 17:54:36 +00:00
bitStream . Write ( playerData - > zoneID . GetCloneID ( ) ) ;
}
2022-07-13 03:36:06 +00:00
bitStream . Write < uint8_t > ( isBestFriend ) ; //isBFF
2021-12-05 17:54:36 +00:00
bitStream . Write < uint8_t > ( 0 ) ; //isFTP
SystemAddress sysAddr = friendData - > sysAddr ;
SEND_PACKET ;
}
2022-07-13 03:36:06 +00:00
void ChatPacketHandler : : SendFriendRequest ( PlayerData * receiver , PlayerData * sender ) {
2021-12-05 17:54:36 +00:00
if ( ! receiver | | ! sender ) return ;
//Make sure people aren't requesting people that they're already friends with:
for ( auto fr : receiver - > friends ) {
if ( fr . friendID = = sender - > playerID ) {
2023-01-22 23:38:47 +00:00
SendFriendResponse ( sender , receiver , eAddFriendResponseType : : ALREADYFRIEND , fr . isBestFriend ) ;
2021-12-05 17:54:36 +00:00
return ; //we have this player as a friend, yeet this function so it doesn't send another request.
}
}
CBITSTREAM ;
2023-09-21 01:06:28 +00:00
BitStreamUtils : : WriteHeader ( bitStream , eConnectionType : : CHAT_INTERNAL , eChatInternalMessageType : : ROUTE_TO_PLAYER ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( receiver - > playerID ) ;
//portion that will get routed:
2023-09-21 01:06:28 +00:00
BitStreamUtils : : WriteHeader ( bitStream , eConnectionType : : CLIENT , eClientMessageType : : ADD_FRIEND_REQUEST ) ;
bitStream . Write ( LUWString ( sender - > playerName . c_str ( ) ) ) ;
2022-07-13 03:36:06 +00:00
bitStream . Write < uint8_t > ( 0 ) ; // This is a BFF flag however this is unused in live and does not have an implementation client side.
2021-12-05 17:54:36 +00:00
SystemAddress sysAddr = receiver - > sysAddr ;
SEND_PACKET ;
}
2023-01-22 23:38:47 +00:00
void ChatPacketHandler : : SendFriendResponse ( PlayerData * receiver , PlayerData * sender , eAddFriendResponseType responseCode , uint8_t isBestFriendsAlready , uint8_t isBestFriendRequest ) {
2021-12-05 17:54:36 +00:00
if ( ! receiver | | ! sender ) return ;
CBITSTREAM ;
2023-09-21 01:06:28 +00:00
BitStreamUtils : : WriteHeader ( bitStream , eConnectionType : : CHAT_INTERNAL , eChatInternalMessageType : : ROUTE_TO_PLAYER ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( receiver - > playerID ) ;
2022-07-13 03:36:06 +00:00
// Portion that will get routed:
2023-09-21 01:06:28 +00:00
BitStreamUtils : : WriteHeader ( bitStream , eConnectionType : : CLIENT , eClientMessageType : : ADD_FRIEND_RESPONSE ) ;
2022-07-13 03:36:06 +00:00
bitStream . Write ( responseCode ) ;
// For all requests besides accepted, write a flag that says whether or not we are already best friends with the receiver.
2023-01-22 23:38:47 +00:00
bitStream . Write < uint8_t > ( responseCode ! = eAddFriendResponseType : : ACCEPTED ? isBestFriendsAlready : sender - > sysAddr ! = UNASSIGNED_SYSTEM_ADDRESS ) ;
2022-07-13 03:36:06 +00:00
// Then write the player name
2023-09-21 01:06:28 +00:00
bitStream . Write ( LUWString ( sender - > playerName . c_str ( ) ) ) ;
2022-07-13 03:36:06 +00:00
// Then if this is an acceptance code, write the following extra info.
2023-01-22 23:38:47 +00:00
if ( responseCode = = eAddFriendResponseType : : ACCEPTED ) {
2022-07-13 03:36:06 +00:00
bitStream . Write ( sender - > playerID ) ;
bitStream . Write ( sender - > zoneID ) ;
bitStream . Write ( isBestFriendRequest ) ; //isBFF
bitStream . Write < uint8_t > ( 0 ) ; //isFTP
}
2021-12-05 17:54:36 +00:00
SystemAddress sysAddr = receiver - > sysAddr ;
SEND_PACKET ;
}
void ChatPacketHandler : : SendRemoveFriend ( PlayerData * receiver , std : : string & personToRemove , bool isSuccessful ) {
if ( ! receiver ) return ;
CBITSTREAM ;
2023-09-21 01:06:28 +00:00
BitStreamUtils : : WriteHeader ( bitStream , eConnectionType : : CHAT_INTERNAL , eChatInternalMessageType : : ROUTE_TO_PLAYER ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write ( receiver - > playerID ) ;
//portion that will get routed:
2023-09-21 01:06:28 +00:00
BitStreamUtils : : WriteHeader ( bitStream , eConnectionType : : CLIENT , eClientMessageType : : REMOVE_FRIEND_RESPONSE ) ;
2021-12-05 17:54:36 +00:00
bitStream . Write < uint8_t > ( isSuccessful ) ; //isOnline
2023-09-21 01:06:28 +00:00
bitStream . Write ( LUWString ( personToRemove ) ) ;
2021-12-05 17:54:36 +00:00
SystemAddress sysAddr = receiver - > sysAddr ;
SEND_PACKET ;
}