psu_proto: Move chat (the one sent by the client) to events.
This commit is contained in:
parent
b52c7ccdac
commit
ecef3e3e8b
@ -524,6 +524,36 @@ event({area_change, QuestID, ZoneID, MapID, EntryID}) ->
|
|||||||
log("area change (~b,~b,~b,~b)", [QuestID, ZoneID, MapID, EntryID]),
|
log("area change (~b,~b,~b,~b)", [QuestID, ZoneID, MapID, EntryID]),
|
||||||
area_load(QuestID, ZoneID, MapID, EntryID);
|
area_load(QuestID, ZoneID, MapID, EntryID);
|
||||||
|
|
||||||
|
%% @doc Chat broadcast handler. Dispatch the message to everyone (for now).
|
||||||
|
%% Disregard the name sent by the server. Use the name saved in memory instead, to prevent client-side editing.
|
||||||
|
%% @todo Only broadcast to people in the same map.
|
||||||
|
%% @todo In the case of NPC characters, when FromTypeID is 00001d00, check that the NPC is in the party and broadcast only to the party (probably).
|
||||||
|
%% @todo When the game doesn't find an NPC (probably) and forces it to talk like in the tutorial mission it seems FromTypeID, FromGID and Name are all 0.
|
||||||
|
%% @todo Make sure modifiers have correct values.
|
||||||
|
event({chat, _FromTypeID, FromGID, _FromName, Modifiers, ChatMsg}) ->
|
||||||
|
UserGID = get(gid),
|
||||||
|
[BcastTypeID, BcastGID, BcastName] = case FromGID of
|
||||||
|
0 -> %% @todo unknown; just fail for now.
|
||||||
|
log("event chat FromGID=0"),
|
||||||
|
ignore; %% @todo This will trigger an error.
|
||||||
|
UserGID -> %% player chat: disregard whatever was sent except modifiers and message.
|
||||||
|
{ok, User} = egs_user_model:read(UserGID),
|
||||||
|
[16#00001200, User#egs_user_model.id, (User#egs_user_model.character)#characters.name];
|
||||||
|
NPCGID -> %% npc chat: @todo Check that the player is the party leader and this npc is in his party.
|
||||||
|
{ok, User} = egs_user_model:read(NPCGID),
|
||||||
|
[16#00001d00, FromGID, (User#egs_user_model.character)#characters.name]
|
||||||
|
end,
|
||||||
|
%% log the message as ascii to the console
|
||||||
|
if FromGID =:= 0 -> ignore; true ->
|
||||||
|
[LogName|_] = re:split(BcastName, "\\0\\0", [{return, binary}]),
|
||||||
|
[TmpMessage|_] = re:split(ChatMsg, "\\0\\0", [{return, binary}]),
|
||||||
|
LogMessage = re:replace(TmpMessage, "\\n", " ", [global, {return, binary}]),
|
||||||
|
log("chat from ~s: ~s", [[re:replace(LogName, "\\0", "", [global, {return, binary}])], [re:replace(LogMessage, "\\0", "", [global, {return, binary}])]])
|
||||||
|
end,
|
||||||
|
%% broadcast
|
||||||
|
{ok, List} = egs_user_model:select(all),
|
||||||
|
lists:foreach(fun(X) -> X#egs_user_model.pid ! {psu_chat, BcastTypeID, BcastGID, BcastName, Modifiers, ChatMsg} end, List);
|
||||||
|
|
||||||
%% @todo Make sure non-mission counters follow the same loading process.
|
%% @todo Make sure non-mission counters follow the same loading process.
|
||||||
%% @todo Probably validate the From* values, to not send the player back inside a mission.
|
%% @todo Probably validate the From* values, to not send the player back inside a mission.
|
||||||
event({counter_enter, CounterID, FromZoneID, FromMapID, FromEntryID}) ->
|
event({counter_enter, CounterID, FromZoneID, FromMapID, FromEntryID}) ->
|
||||||
@ -809,31 +839,6 @@ event({unicube_select, Selection, EntryID}) ->
|
|||||||
handle(16#0102, _) ->
|
handle(16#0102, _) ->
|
||||||
ignore;
|
ignore;
|
||||||
|
|
||||||
%% @doc Chat broadcast handler. Dispatch the message to everyone (for now).
|
|
||||||
%% Disregard the name sent by the server. Use the name saved in memory instead, to prevent client-side editing.
|
|
||||||
%% @todo Only broadcast to people in the same map.
|
|
||||||
%% @todo In the case of NPC characters, when FromTypeID is 00001d00, check that the NPC is in the party and broadcast only to the party (probably).
|
|
||||||
%% @todo When the game doesn't find an NPC and forces it to talk like in the tutorial mission it seems FromTypeID, FromGID and Name are both 0.
|
|
||||||
handle(16#0304, Data) ->
|
|
||||||
<< FromTypeID:32/unsigned-integer, FromGID:32/little-unsigned-integer, Modifiers:128/bits, _:512, Message/bits >> = Data,
|
|
||||||
|
|
||||||
UserGID = get(gid),
|
|
||||||
GID = if UserGID =:= FromGID ->
|
|
||||||
UserGID;
|
|
||||||
true ->
|
|
||||||
%% @todo Check that FromGID is an NPC in the UserGID's party; that UserGID is the party leader; that the message is using party chat.
|
|
||||||
FromGID
|
|
||||||
end,
|
|
||||||
|
|
||||||
{ok, User} = egs_user_model:read(GID),
|
|
||||||
|
|
||||||
[LogName|_] = re:split((User#egs_user_model.character)#characters.name, "\\0\\0", [{return, binary}]),
|
|
||||||
[TmpMessage|_] = re:split(Message, "\\0\\0", [{return, binary}]),
|
|
||||||
LogMessage = re:replace(TmpMessage, "\\n", " ", [global, {return, binary}]),
|
|
||||||
log("chat from ~s: ~s", [[re:replace(LogName, "\\0", "", [global, {return, binary}])], [re:replace(LogMessage, "\\0", "", [global, {return, binary}])]]),
|
|
||||||
{ok, List} = egs_user_model:select(all),
|
|
||||||
lists:foreach(fun(X) -> X#egs_user_model.pid ! {psu_chat, FromTypeID, GID, (User#egs_user_model.character)#characters.name, Modifiers, Message} end, List);
|
|
||||||
|
|
||||||
%% @todo Handle this packet properly.
|
%% @todo Handle this packet properly.
|
||||||
%% @todo Spawn cleared response event shouldn't be handled following this packet but when we see the spawn actually dead HP-wise.
|
%% @todo Spawn cleared response event shouldn't be handled following this packet but when we see the spawn actually dead HP-wise.
|
||||||
handle(16#0402, Data) ->
|
handle(16#0402, Data) ->
|
||||||
@ -1280,7 +1285,9 @@ send_0236() ->
|
|||||||
|
|
||||||
%% @doc Send a chat command.
|
%% @doc Send a chat command.
|
||||||
send_0304(FromTypeID, FromGID, FromName, Modifiers, Message) ->
|
send_0304(FromTypeID, FromGID, FromName, Modifiers, Message) ->
|
||||||
send(<< 16#03040300:32, 0:288, FromTypeID:32/unsigned-integer, FromGID:32/little-unsigned-integer, Modifiers:128/bits, FromName:512/bits, Message/bits >>).
|
{chat_modifiers, ChatType, ChatCutIn, ChatCutInAngle, ChatMsgLength, ChatChannel, ChatCharacterType} = Modifiers,
|
||||||
|
send(<< 16#03040300:32, 0:288, FromTypeID:32/unsigned-integer, FromGID:32/little-unsigned-integer, 0:64,
|
||||||
|
ChatType:8, ChatCutIn:8, ChatCutInAngle:8, ChatMsgLength:8, ChatChannel:8, ChatCharacterType:8, 0:16, FromName:512/bits, Message/bits >>).
|
||||||
|
|
||||||
%% @todo Force send a new player location. Used for warps.
|
%% @todo Force send a new player location. Used for warps.
|
||||||
%% @todo The value before IntDir seems to be the player's current animation. 01 stand up, 08 ?, 17 normal sit
|
%% @todo The value before IntDir seems to be the player's current animation. 01 stand up, 08 ?, 17 normal sit
|
||||||
|
@ -194,6 +194,27 @@ parse(Size, 16#021f, Channel, Data) ->
|
|||||||
end,
|
end,
|
||||||
{unicube_select, Selection, EntryID};
|
{unicube_select, Selection, EntryID};
|
||||||
|
|
||||||
|
parse(_Size, 16#0304, Channel, Data) ->
|
||||||
|
<< _LID:16/little, VarB:16/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little,
|
||||||
|
VarG:32/little, VarH:32/little, VarI:32/little, VarJ:32/little, FromTypeID:32, FromGID:32/little,
|
||||||
|
VarK:32/little, VarL:32/little, ChatType:8, ChatCutIn:8, ChatCutInAngle:8, ChatMsgLength:8,
|
||||||
|
ChatChannel:8, ChatCharacterType:8, VarN:8, _VarO:8, FromName:512, ChatMsg/bits >> = Data,
|
||||||
|
?ASSERT_EQ(Channel, 2),
|
||||||
|
?ASSERT_EQ(VarB, 0),
|
||||||
|
?ASSERT_EQ(VarC, 0),
|
||||||
|
?ASSERT_EQ(VarD, 0),
|
||||||
|
?ASSERT_EQ(VarE, 0),
|
||||||
|
?ASSERT_EQ(VarF, 0),
|
||||||
|
?ASSERT_EQ(VarG, 0),
|
||||||
|
?ASSERT_EQ(VarH, 0),
|
||||||
|
?ASSERT_EQ(VarI, 0),
|
||||||
|
?ASSERT_EQ(VarJ, 0),
|
||||||
|
?ASSERT_EQ(VarK, 0),
|
||||||
|
?ASSERT_EQ(VarL, 0),
|
||||||
|
?ASSERT_EQ(VarN, 0),
|
||||||
|
Modifiers = {chat_modifiers, ChatType, ChatCutIn, ChatCutInAngle, ChatMsgLength, ChatChannel, ChatCharacterType},
|
||||||
|
{chat, FromTypeID, FromGID, FromName, Modifiers, ChatMsg};
|
||||||
|
|
||||||
parse(Size, 16#0807, Channel, Data) ->
|
parse(Size, 16#0807, Channel, Data) ->
|
||||||
<< _LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little,
|
<< _LID:16/little, VarA:16/little, VarB:32/little, VarC:32/little, VarD:32/little, VarE:32/little, VarF:32/little, VarG:32/little, VarH:32/little, VarI:32/little,
|
||||||
QuestID:32/little, ZoneID:16/little, MapID:16/little, EntryID:16/little, _AreaChangeNb:16/little, VarJ:32/little >> = Data,
|
QuestID:32/little, ZoneID:16/little, MapID:16/little, EntryID:16/little, _AreaChangeNb:16/little, VarJ:32/little >> = Data,
|
||||||
|
Loading…
Reference in New Issue
Block a user