psu_game: Remove boring blank lines.

This commit is contained in:
Loïc Hoguin 2010-07-30 23:05:13 +02:00
parent b3bcdad6ca
commit 8495048240

View File

@ -50,13 +50,11 @@ cleanup(Pid) ->
end. end.
%% @doc Listen for connections. %% @doc Listen for connections.
listen(Port, MPid) -> listen(Port, MPid) ->
{ok, LSocket} = ssl:listen(Port, ?OPTIONS), {ok, LSocket} = ssl:listen(Port, ?OPTIONS),
?MODULE:accept(LSocket, MPid). ?MODULE:accept(LSocket, MPid).
%% @doc Accept connections. %% @doc Accept connections.
accept(LSocket, MPid) -> accept(LSocket, MPid) ->
case ssl:transport_accept(LSocket, 5000) of case ssl:transport_accept(LSocket, 5000) of
{ok, CSocket} -> {ok, CSocket} ->
@ -69,7 +67,6 @@ accept(LSocket, MPid) ->
?MODULE:accept(LSocket, MPid). ?MODULE:accept(LSocket, MPid).
%% @doc Initialize the client process by saving the socket to the process dictionary. %% @doc Initialize the client process by saving the socket to the process dictionary.
process_init(CSocket, MPid) -> process_init(CSocket, MPid) ->
link(MPid), link(MPid),
put(socket, CSocket), put(socket, CSocket),
@ -79,7 +76,6 @@ process_init(CSocket, MPid) ->
%% @doc Process the new connections. %% @doc Process the new connections.
%% Send an hello packet, authenticate the user and send him to character select. %% Send an hello packet, authenticate the user and send him to character select.
process() -> process() ->
case egs_proto:packet_recv(get(socket), 5000) of case egs_proto:packet_recv(get(socket), 5000) of
{ok, Orig} -> {ok, Orig} ->
@ -93,7 +89,6 @@ process() ->
end. end.
%% @doc Game server auth request handler. Save the GID in the process dictionary after checking it. %% @doc Game server auth request handler. Save the GID in the process dictionary after checking it.
process_handle(16#020d, << GID:32/little-unsigned-integer, Auth:32/bits, _/bits >>) -> process_handle(16#020d, << GID:32/little-unsigned-integer, Auth:32/bits, _/bits >>) ->
CSocket = get(socket), CSocket = get(socket),
case egs_user_model:read(GID) of case egs_user_model:read(GID) of
@ -118,19 +113,16 @@ process_handle(16#020d, << GID:32/little-unsigned-integer, Auth:32/bits, _/bits
end; end;
%% @doc Platform information handler. Obtain the game version and save it into the process dictionary. %% @doc Platform information handler. Obtain the game version and save it into the process dictionary.
process_handle(16#080e, << _:64, Version:32/little-unsigned-integer, _/bits >>) -> process_handle(16#080e, << _:64, Version:32/little-unsigned-integer, _/bits >>) ->
put(version, Version), put(version, Version),
?MODULE:process(); ?MODULE:process();
%% @doc Unknown command handler. Do nothing. %% @doc Unknown command handler. Do nothing.
process_handle(Command, _) -> process_handle(Command, _) ->
log("(process) dismissed packet ~4.16.0b", [Command]), log("(process) dismissed packet ~4.16.0b", [Command]),
?MODULE:process(). ?MODULE:process().
%% @doc Character selection screen loop. %% @doc Character selection screen loop.
char_select() -> char_select() ->
case egs_proto:packet_recv(get(socket), 5000) of case egs_proto:packet_recv(get(socket), 5000) of
{ok, Orig} -> {ok, Orig} ->
@ -145,7 +137,6 @@ char_select() ->
end. end.
%% @doc Character selection handler. %% @doc Character selection handler.
char_select_handle(16#020b, << Number:32/little-unsigned-integer, _/bits >>) -> char_select_handle(16#020b, << Number:32/little-unsigned-integer, _/bits >>) ->
log("selected character ~b", [Number]), log("selected character ~b", [Number]),
char_select_load(Number); char_select_load(Number);
@ -168,26 +159,22 @@ char_select_handle(16#0d02, << Number:32/little-unsigned-integer, Char/bits >>)
char_select_load(Number); char_select_load(Number);
%% @doc Character selection screen request. %% @doc Character selection screen request.
char_select_handle(16#0d06, _) -> char_select_handle(16#0d06, _) ->
{ok, User} = egs_user_model:read(get(gid)), {ok, User} = egs_user_model:read(get(gid)),
send_0d03(data_load(User#egs_user_model.folder, 0), data_load(User#egs_user_model.folder, 1), data_load(User#egs_user_model.folder, 2), data_load(User#egs_user_model.folder, 3)), send_0d03(data_load(User#egs_user_model.folder, 0), data_load(User#egs_user_model.folder, 1), data_load(User#egs_user_model.folder, 2), data_load(User#egs_user_model.folder, 3)),
?MODULE:char_select(); ?MODULE:char_select();
%% @doc Silently ignore packet 0818. Gives CPU/GPU information. %% @doc Silently ignore packet 0818. Gives CPU/GPU information.
char_select_handle(16#0818, _) -> char_select_handle(16#0818, _) ->
?MODULE:char_select(); ?MODULE:char_select();
%% @doc Unknown command handler. Do nothing. %% @doc Unknown command handler. Do nothing.
char_select_handle(Command, _) -> char_select_handle(Command, _) ->
log("(char_select) dismissed packet ~4.16.0b", [Command]), log("(char_select) dismissed packet ~4.16.0b", [Command]),
?MODULE:char_select(). ?MODULE:char_select().
%% @doc Load the selected character in the start lobby and start the main game's loop. %% @doc Load the selected character in the start lobby and start the main game's loop.
%% The default entry point currently is 4th floor, Linear Line counter. %% The default entry point currently is 4th floor, Linear Line counter.
char_select_load(Number) -> char_select_load(Number) ->
{ok, OldUser} = egs_user_model:read(get(gid)), {ok, OldUser} = egs_user_model:read(get(gid)),
[{status, 1}, {char, CharBin}, {options, OptionsBin}] = data_load(OldUser#egs_user_model.folder, Number), [{status, 1}, {char, CharBin}, {options, OptionsBin}] = data_load(OldUser#egs_user_model.folder, Number),
@ -208,7 +195,6 @@ char_select_load(Number) ->
?MODULE:loop(<< >>). ?MODULE:loop(<< >>).
%% @doc Load the given character's data. %% @doc Load the given character's data.
data_load(Folder, Number) -> data_load(Folder, Number) ->
Filename = io_lib:format("save/~s/~b-character", [Folder, Number]), Filename = io_lib:format("save/~s/~b-character", [Folder, Number]),
case file:read_file(Filename) of case file:read_file(Filename) of
@ -220,7 +206,6 @@ data_load(Folder, Number) ->
end. end.
%% @doc Load and send the character information to the client. %% @doc Load and send the character information to the client.
char_load(User) -> char_load(User) ->
send_0d01(User), send_0d01(User),
% 0246 % 0246
@ -237,7 +222,6 @@ char_load(User) ->
send_1602(). send_1602().
%% @doc Load the given map as a mission counter. %% @doc Load the given map as a mission counter.
counter_load(QuestID, ZoneID, MapID, EntryID) -> counter_load(QuestID, ZoneID, MapID, EntryID) ->
{ok, OldUser} = egs_user_model:read(get(gid)), {ok, OldUser} = egs_user_model:read(get(gid)),
OldArea = OldUser#egs_user_model.area, OldArea = OldUser#egs_user_model.area,
@ -273,7 +257,6 @@ counter_load(QuestID, ZoneID, MapID, EntryID) ->
send_0236(). send_0236().
%% @doc Return the current season information. %% @doc Return the current season information.
area_get_season(QuestID) -> area_get_season(QuestID) ->
{{_, Month, Day}, _} = calendar:universal_time(), {{_, Month, Day}, _} = calendar:universal_time(),
[IsSeasonal, SeasonID, SeasonQuestIDs] = if [IsSeasonal, SeasonID, SeasonQuestIDs] = if
@ -313,7 +296,6 @@ area_get_season(QuestID) ->
end. end.
%% @doc Load the given map as a standard lobby. %% @doc Load the given map as a standard lobby.
area_load(QuestID, ZoneID, MapID, EntryID) -> area_load(QuestID, ZoneID, MapID, EntryID) ->
{ok, OldUser} = egs_user_model:read(get(gid)), {ok, OldUser} = egs_user_model:read(get(gid)),
[{type, AreaType}, {file, QuestFile}|MissionInfo] = proplists:get_value(QuestID, ?QUESTS, [{type, undefined}, {file, undefined}]), [{type, AreaType}, {file, QuestFile}|MissionInfo] = proplists:get_value(QuestID, ?QUESTS, [{type, undefined}, {file, undefined}]),
@ -445,7 +427,6 @@ myroom_send_packet(Filename) ->
%% @doc Game's main loop. %% @doc Game's main loop.
%% @todo We probably don't want to send a keepalive packet unnecessarily. %% @todo We probably don't want to send a keepalive packet unnecessarily.
loop(SoFar) -> loop(SoFar) ->
receive receive
{psu_broadcast, Orig} -> {psu_broadcast, Orig} ->
@ -487,7 +468,6 @@ loop(SoFar) ->
end. end.
%% @doc Dispatch the command to the right handler. %% @doc Dispatch the command to the right handler.
dispatch(Orig) -> dispatch(Orig) ->
{command, Command, Channel, Data} = egs_proto:packet_parse(Orig), {command, Command, Channel, Data} = egs_proto:packet_parse(Orig),
case Channel of case Channel of
@ -497,7 +477,6 @@ dispatch(Orig) ->
end. end.
%% @doc Position change broadcast handler. Save the position and then dispatch it. %% @doc Position change broadcast handler. Save the position and then dispatch it.
broadcast(16#0503, Orig) -> broadcast(16#0503, Orig) ->
<< _:424, Dir:24/little-unsigned-integer, _PrevCoords:96, X:32/little-float, Y:32/little-float, Z:32/little-float, << _:424, Dir:24/little-unsigned-integer, _PrevCoords:96, X:32/little-float, Y:32/little-float, Z:32/little-float,
QuestID:32/little-unsigned-integer, ZoneID:32/little-unsigned-integer, MapID:32/little-unsigned-integer, EntryID:32/little-unsigned-integer, _:32 >> = Orig, QuestID:32/little-unsigned-integer, ZoneID:32/little-unsigned-integer, MapID:32/little-unsigned-integer, EntryID:32/little-unsigned-integer, _:32 >> = Orig,
@ -508,7 +487,6 @@ broadcast(16#0503, Orig) ->
broadcast(default, Orig); broadcast(default, Orig);
%% @doc Stand still broadcast handler. Save the position and then dispatch it. %% @doc Stand still broadcast handler. Save the position and then dispatch it.
broadcast(16#0514, Orig) -> broadcast(16#0514, Orig) ->
<< _:424, Dir:24/little-unsigned-integer, X:32/little-float, Y:32/little-float, Z:32/little-float, << _:424, Dir:24/little-unsigned-integer, X:32/little-float, Y:32/little-float, Z:32/little-float,
QuestID:32/little-unsigned-integer, ZoneID:32/little-unsigned-integer, QuestID:32/little-unsigned-integer, ZoneID:32/little-unsigned-integer,
@ -523,7 +501,6 @@ broadcast(16#0514, Orig) ->
%% We clean up the command and use the real GID and LID of the user, disregarding what was sent and possibly tampered with. %% We clean up the command and use the real GID and LID of the user, disregarding what was sent and possibly tampered with.
%% Only a handful of commands are allowed to broadcast. An user tampering with it would get disconnected instantly. %% Only a handful of commands are allowed to broadcast. An user tampering with it would get disconnected instantly.
%% @todo Don't query the user data everytime! Keep an User instead of a GID probably. %% @todo Don't query the user data everytime! Keep an User instead of a GID probably.
broadcast(Command, Orig) broadcast(Command, Orig)
when Command =:= 16#0101; when Command =:= 16#0101;
Command =:= 16#0102; Command =:= 16#0102;
@ -546,7 +523,6 @@ broadcast(Command, Orig)
end. end.
%% @doc Movement (non-broadcast) handler. Do nothing. %% @doc Movement (non-broadcast) handler. Do nothing.
handle(16#0102, _) -> handle(16#0102, _) ->
ignore; ignore;
@ -557,7 +533,6 @@ handle(16#0102, _) ->
%% @todo Others probably want to see that you changed your weapon. %% @todo Others probably want to see that you changed your weapon.
%% @todo Apparently B is always ItemID+1. Not sure why. %% @todo Apparently B is always ItemID+1. Not sure why.
%% @todo Currently use a separate file for the data sent for the weapons. %% @todo Currently use a separate file for the data sent for the weapons.
handle(16#0105, Data) -> handle(16#0105, Data) ->
<< _:32, A:32/little-unsigned-integer, ItemID:8, Action:8, _:8, B:8, C:32/little-unsigned-integer, _/bits >> = Data, << _:32, A:32/little-unsigned-integer, ItemID:8, Action:8, _:8, B:8, C:32/little-unsigned-integer, _/bits >> = Data,
log("0105 action ~b item ~b (~b ~b ~b)", [Action, ItemID, A, B, C]), log("0105 action ~b item ~b (~b ~b ~b)", [Action, ItemID, A, B, C]),
@ -606,7 +581,6 @@ handle(16#0105, Data) ->
%% @doc Shop listing request. Currently return the normal item shop for everything. %% @doc Shop listing request. Currently return the normal item shop for everything.
%% @todo Return the other shops appropriately. %% @todo Return the other shops appropriately.
handle(16#010a, Data) -> handle(16#010a, Data) ->
<< _:32, A:32/little-unsigned-integer, B:32/little-unsigned-integer, C:32/little-unsigned-integer >> = Data, << _:32, A:32/little-unsigned-integer, B:32/little-unsigned-integer, C:32/little-unsigned-integer >> = Data,
log("shop listing request (~b, ~b, ~b)", [A, B, C]), log("shop listing request (~b, ~b, ~b)", [A, B, C]),
@ -617,7 +591,6 @@ handle(16#010a, Data) ->
%% @doc Character death, and more, handler. Warp to 4th floor for now. %% @doc Character death, and more, handler. Warp to 4th floor for now.
%% @todo Recover from death correctly. %% @todo Recover from death correctly.
handle(16#0110, Data) -> handle(16#0110, Data) ->
<< _:32, A:32/little-unsigned-integer, B:32/little-unsigned-integer, C:32/little-unsigned-integer >> = Data, << _:32, A:32/little-unsigned-integer, B:32/little-unsigned-integer, C:32/little-unsigned-integer >> = Data,
case B of case B of
@ -643,7 +616,6 @@ handle(16#0110, Data) ->
end; end;
%% @doc Uni cube handler. %% @doc Uni cube handler.
handle(16#021d, _) -> handle(16#021d, _) ->
send_021e(); send_021e();
@ -651,7 +623,6 @@ handle(16#021d, _) ->
%% When selecting 'Your room', load a default room. %% When selecting 'Your room', load a default room.
%% When selecting 'Reload', reload the character in the current lobby. %% When selecting 'Reload', reload the character in the current lobby.
%% @todo There's probably an entryid given in the uni selection packet. %% @todo There's probably an entryid given in the uni selection packet.
handle(16#021f, << Uni:32/little-unsigned-integer, _/bits >>) -> handle(16#021f, << Uni:32/little-unsigned-integer, _/bits >>) ->
case Uni of case Uni of
0 -> % cancelled uni selection 0 -> % cancelled uni selection
@ -675,7 +646,6 @@ handle(16#021f, << Uni:32/little-unsigned-integer, _/bits >>) ->
%% @doc Shortcut changes handler. Do nothing. %% @doc Shortcut changes handler. Do nothing.
%% @todo Save it. %% @todo Save it.
handle(16#0302, _) -> handle(16#0302, _) ->
log("dismissed shortcut changes"); log("dismissed shortcut changes");
@ -683,7 +653,6 @@ handle(16#0302, _) ->
%% We must take extra precautions to handle different versions of the game correctly. %% We must take extra precautions to handle different versions of the game correctly.
%% Disregard the name sent by the server in later versions of the game. Use the name saved in memory instead, to prevent client-side editing. %% Disregard the name sent by the server in later versions of the game. Use the name saved in memory instead, to prevent client-side editing.
%% @todo Only broadcast to people in the same map. %% @todo Only broadcast to people in the same map.
handle(16#0304, Data) -> handle(16#0304, Data) ->
{ok, User} = egs_user_model:read(get(gid)), {ok, User} = egs_user_model:read(get(gid)),
case get(version) of case get(version) of
@ -701,7 +670,6 @@ handle(16#0304, Data) ->
%% @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) ->
<< SpawnID:32/little-unsigned-integer, _:64, Type:32/little-unsigned-integer, _:64 >> = Data, << SpawnID:32/little-unsigned-integer, _:64, Type:32/little-unsigned-integer, _:64 >> = Data,
case Type of case Type of
@ -718,7 +686,6 @@ handle(16#0402, Data) ->
%% @todo Handle this packet. %% @todo Handle this packet.
%% @todo 3rd Unsafe Passage C, EventID 10 BlockID 2 = mission cleared? %% @todo 3rd Unsafe Passage C, EventID 10 BlockID 2 = mission cleared?
handle(16#0404, Data) -> handle(16#0404, Data) ->
<< EventID:8, BlockID:8, _:16, Value:8, _/bits >> = Data, << EventID:8, BlockID:8, _:16, Value:8, _/bits >> = Data,
log("unknown command 0404: eventid ~b blockid ~b value ~b", [EventID, BlockID, Value]), log("unknown command 0404: eventid ~b blockid ~b value ~b", [EventID, BlockID, Value]),
@ -727,7 +694,6 @@ handle(16#0404, Data) ->
%% @doc Map change handler. %% @doc Map change handler.
%% Rooms are handled differently than normal lobbies. %% Rooms are handled differently than normal lobbies.
%% @todo When changing lobby to the room, 0230 must also be sent. Same when going from room to lobby. %% @todo When changing lobby to the room, 0230 must also be sent. Same when going from room to lobby.
handle(16#0807, Data) -> handle(16#0807, Data) ->
<< QuestID:32/little-unsigned-integer, ZoneID:16/little-unsigned-integer, << QuestID:32/little-unsigned-integer, ZoneID:16/little-unsigned-integer,
MapID:16/little-unsigned-integer, EntryID:16/little-unsigned-integer, _/bits >> = Data, MapID:16/little-unsigned-integer, EntryID:16/little-unsigned-integer, _/bits >> = Data,
@ -735,7 +701,6 @@ handle(16#0807, Data) ->
area_load(QuestID, ZoneID, MapID, EntryID); area_load(QuestID, ZoneID, MapID, EntryID);
%% @doc Mission counter handler. %% @doc Mission counter handler.
handle(16#0811, Data) -> handle(16#0811, Data) ->
<< QuestID:32/little-unsigned-integer, ZoneID:16/little-unsigned-integer, << QuestID:32/little-unsigned-integer, ZoneID:16/little-unsigned-integer,
MapID:16/little-unsigned-integer, EntryID:16/little-unsigned-integer, _/bits >> = Data, MapID:16/little-unsigned-integer, EntryID:16/little-unsigned-integer, _/bits >> = Data,
@ -743,7 +708,6 @@ handle(16#0811, Data) ->
counter_load(QuestID, ZoneID, MapID, EntryID); counter_load(QuestID, ZoneID, MapID, EntryID);
%% @doc Leave mission counter handler. Lobby values depend on which counter was entered. %% @doc Leave mission counter handler. Lobby values depend on which counter was entered.
handle(16#0812, _) -> handle(16#0812, _) ->
{ok, User} = egs_user_model:read(get(gid)), {ok, User} = egs_user_model:read(get(gid)),
Area = User#egs_user_model.area, Area = User#egs_user_model.area,
@ -766,13 +730,11 @@ handle(16#0813, Data) ->
%% @doc Item description request. %% @doc Item description request.
%% @todo Send something other than just "dammy". %% @todo Send something other than just "dammy".
handle(16#0a10, << ItemID:32/unsigned-integer >>) -> handle(16#0a10, << ItemID:32/unsigned-integer >>) ->
send_0a11(ItemID, "dammy"); send_0a11(ItemID, "dammy");
%% @doc Start mission handler. %% @doc Start mission handler.
%% @todo Forward the mission start to other players of the same party, whatever their location is. %% @todo Forward the mission start to other players of the same party, whatever their location is.
handle(16#0c01, << QuestID:32/little-unsigned-integer >>) -> handle(16#0c01, << QuestID:32/little-unsigned-integer >>) ->
log("start mission ~b", [QuestID]), log("start mission ~b", [QuestID]),
send_170c(), send_170c(),
@ -782,7 +744,6 @@ handle(16#0c01, << QuestID:32/little-unsigned-integer >>) ->
%% @doc Counter quests files request handler? Send huge number of quest files. %% @doc Counter quests files request handler? Send huge number of quest files.
%% @todo Handle correctly. %% @todo Handle correctly.
handle(16#0c05, _) -> handle(16#0c05, _) ->
{ok, User} = egs_user_model:read(get(gid)), {ok, User} = egs_user_model:read(get(gid)),
[{quests, Filename}, {bg, _}, {options, _}] = proplists:get_value(User#egs_user_model.entryid, ?COUNTERS), [{quests, Filename}, {bg, _}, {options, _}] = proplists:get_value(User#egs_user_model.entryid, ?COUNTERS),
@ -790,13 +751,11 @@ handle(16#0c05, _) ->
%% @doc Lobby transport handler? Just ignore the meseta price for now and send the player where he wanna be! %% @doc Lobby transport handler? Just ignore the meseta price for now and send the player where he wanna be!
%% @todo Handle correctly. %% @todo Handle correctly.
handle(16#0c07, _) -> handle(16#0c07, _) ->
send_0c08(true); send_0c08(true);
%% @doc Abort mission handler. %% @doc Abort mission handler.
%% Replenish the player HP. %% Replenish the player HP.
handle(16#0c0e, _) -> handle(16#0c0e, _) ->
send_1006(11), send_1006(11),
{ok, User} = egs_user_model:read(get(gid)), {ok, User} = egs_user_model:read(get(gid)),
@ -817,7 +776,6 @@ handle(16#0c0e, _) ->
%% @doc Counter available mission list request handler. %% @doc Counter available mission list request handler.
%% @todo Temporarily allow rare mission and LL all difficulties to all players. %% @todo Temporarily allow rare mission and LL all difficulties to all players.
handle(16#0c0f, _) -> handle(16#0c0f, _) ->
{ok, User} = egs_user_model:read(get(gid)), {ok, User} = egs_user_model:read(get(gid)),
[{quests, _}, {bg, _}, {options, Options}] = proplists:get_value(User#egs_user_model.entryid, ?COUNTERS), [{quests, _}, {bg, _}, {options, Options}] = proplists:get_value(User#egs_user_model.entryid, ?COUNTERS),
@ -826,7 +784,6 @@ handle(16#0c0f, _) ->
%% @doc Set flag handler. Associate a new flag with the character. %% @doc Set flag handler. Associate a new flag with the character.
%% Just reply with a success value for now. %% Just reply with a success value for now.
%% @todo God save the flags. %% @todo God save the flags.
handle(16#0d04, Data) -> handle(16#0d04, Data) ->
<< Flag:128/bits, A:16/bits, _:8, B/bits >> = Data, << Flag:128/bits, A:16/bits, _:8, B/bits >> = Data,
log("flag handler for ~s", [re:replace(Flag, "\\0+", "", [global, {return, binary}])]), log("flag handler for ~s", [re:replace(Flag, "\\0+", "", [global, {return, binary}])]),
@ -834,7 +791,6 @@ handle(16#0d04, Data) ->
send(<< 16#0d040300:32, 0:160, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, Flag/binary, A/binary, 1, B/binary >>); send(<< 16#0d040300:32, 0:160, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, Flag/binary, A/binary, 1, B/binary >>);
%% @doc Options changes handler. %% @doc Options changes handler.
handle(16#0d07, Data) -> handle(16#0d07, Data) ->
log("options changes"), log("options changes"),
% Translate options into a tuple, validate them and do nothing with it for now % Translate options into a tuple, validate them and do nothing with it for now
@ -847,7 +803,6 @@ handle(16#0d07, Data) ->
%% @doc Hit handler. %% @doc Hit handler.
%% @todo Finish the work on it. %% @todo Finish the work on it.
%% @todo First value at 2C is the number of hits. We don't need to know it though. %% @todo First value at 2C is the number of hits. We don't need to know it though.
handle(16#0e00, Data) -> handle(16#0e00, Data) ->
<< _:96, Hits/bits >> = Data, << _:96, Hits/bits >> = Data,
handle_hits(Hits); handle_hits(Hits);
@ -881,7 +836,6 @@ handle(16#0f07, Data) ->
%% @doc Object event handler. %% @doc Object event handler.
%% @todo Handle all events appropriately. %% @todo Handle all events appropriately.
%% @todo B should be the ObjType. %% @todo B should be the ObjType.
handle(16#0f0a, Data) -> handle(16#0f0a, Data) ->
<< BlockID:16/little-unsigned-integer, ListNb:16/little-unsigned-integer, ObjectNb:16/little-unsigned-integer, _MapID:16/little-unsigned-integer, ObjectID:16/little-unsigned-integer, << BlockID:16/little-unsigned-integer, ListNb:16/little-unsigned-integer, ObjectNb:16/little-unsigned-integer, _MapID:16/little-unsigned-integer, ObjectID:16/little-unsigned-integer,
_:16, A:32/little-unsigned-integer, B:32/little-unsigned-integer, _:32, C:32/little-unsigned-integer, _:272, Action:8, _/bits >> = Data, _:16, A:32/little-unsigned-integer, B:32/little-unsigned-integer, _:32, C:32/little-unsigned-integer, _:272, Action:8, _/bits >> = Data,
@ -963,12 +917,10 @@ handle(16#1019, Data) ->
%~ send(<< (header(16#1019))/binary, 0:192, 16#00200000:32, 0:32 >>); %~ send(<< (header(16#1019))/binary, 0:192, 16#00200000:32, 0:32 >>);
%% @todo Not sure about that one though. Probably related to 1112 still. %% @todo Not sure about that one though. Probably related to 1112 still.
handle(16#1106, Data) -> handle(16#1106, Data) ->
send_110e(Data); send_110e(Data);
%% @doc Probably asking permission to start the video (used for syncing?). %% @doc Probably asking permission to start the video (used for syncing?).
handle(16#1112, Data) -> handle(16#1112, Data) ->
send_1113(Data); send_1113(Data);
@ -980,31 +932,26 @@ handle(16#1216, Data) ->
%% @doc Party information recap request. %% @doc Party information recap request.
%% @todo Handle when the party already exists! And stop doing it wrong. %% @todo Handle when the party already exists! And stop doing it wrong.
handle(16#1705, _) -> handle(16#1705, _) ->
{ok, User} = egs_user_model:read(get(gid)), {ok, User} = egs_user_model:read(get(gid)),
send_1706((User#egs_user_model.character)#characters.name); send_1706((User#egs_user_model.character)#characters.name);
%% @doc Mission selected handler. Send the currently selected mission. %% @doc Mission selected handler. Send the currently selected mission.
%% @todo Probably need to dispatch that info to other party members in the same counter. %% @todo Probably need to dispatch that info to other party members in the same counter.
handle(16#1707, _) -> handle(16#1707, _) ->
ignore; ignore;
%% @doc Party settings request handler. Item distribution is random for now. %% @doc Party settings request handler. Item distribution is random for now.
%% @todo Handle correctly. %% @todo Handle correctly.
handle(16#1709, _) -> handle(16#1709, _) ->
send_170a(); send_170a();
%% @doc Counter-related handler. %% @doc Counter-related handler.
handle(16#170b, _) -> handle(16#170b, _) ->
send_170c(); send_170c();
%% @doc Counter initialization handler? Send the code for the background image to use. %% @doc Counter initialization handler? Send the code for the background image to use.
%% @todo Handle correctly. %% @todo Handle correctly.
handle(16#1710, _) -> handle(16#1710, _) ->
{ok, User} = egs_user_model:read(get(gid)), {ok, User} = egs_user_model:read(get(gid)),
[{quests, _}, {bg, Background}, {options, _}] = proplists:get_value(User#egs_user_model.entryid, ?COUNTERS), [{quests, _}, {bg, Background}, {options, _}] = proplists:get_value(User#egs_user_model.entryid, ?COUNTERS),
@ -1012,7 +959,6 @@ handle(16#1710, _) ->
%% @doc Dialog request handler. Do what we can. %% @doc Dialog request handler. Do what we can.
%% @todo Handle correctly. %% @todo Handle correctly.
handle(16#1a01, Data) -> handle(16#1a01, Data) ->
<< _:32, A:8, B:8, _:16, C:8, _/bits >> = Data, << _:32, A:8, B:8, _:16, C:8, _/bits >> = Data,
case [A, B, C] of case [A, B, C] of
@ -1048,7 +994,6 @@ handle(16#1a01, Data) ->
end; end;
%% @doc Unknown command handler. Do nothing. %% @doc Unknown command handler. Do nothing.
handle(Command, _) -> handle(Command, _) ->
log("dismissed packet ~4.16.0b", [Command]). log("dismissed packet ~4.16.0b", [Command]).
@ -1106,7 +1051,6 @@ handle_hits(Data) ->
handle_hits(Rest). handle_hits(Rest).
%% @doc Handle a list of events. %% @doc Handle a list of events.
handle_events([]) -> handle_events([]) ->
ok; ok;
handle_events([{explode, ObjectID}|Tail]) -> handle_events([{explode, ObjectID}|Tail]) ->
@ -1117,21 +1061,18 @@ handle_events([{event, [BlockID, EventID]}|Tail]) ->
handle_events(Tail). handle_events(Tail).
%% @doc Build the packet header. %% @doc Build the packet header.
header(Command) -> header(Command) ->
GID = get(gid), GID = get(gid),
<< Command:16/unsigned-integer, 16#0300:16, 0:160, 16#00011300:32, GID:32/little-unsigned-integer, 0:64 >>. << Command:16/unsigned-integer, 16#0300:16, 0:160, 16#00011300:32, GID:32/little-unsigned-integer, 0:64 >>.
%% @doc Send the given packet to the client. %% @doc Send the given packet to the client.
%% @todo Consolidate the receive and send functions better. %% @todo Consolidate the receive and send functions better.
send(Packet) -> send(Packet) ->
egs_proto:packet_send(get(socket), Packet). egs_proto:packet_send(get(socket), Packet).
%% @todo Figure out what this does compared to 0201(self). %% @todo Figure out what this does compared to 0201(self).
%% @todo Figure out the unknown values. %% @todo Figure out the unknown values.
%% @todo Probably don't pattern match the data like this... %% @todo Probably don't pattern match the data like this...
send_010d(User) -> send_010d(User) ->
GID = get(gid), GID = get(gid),
CharGID = User#egs_user_model.id, CharGID = User#egs_user_model.id,
@ -1141,21 +1082,18 @@ send_010d(User) ->
0:192, CharGID:32/little-unsigned-integer, 0:32, 16#ffffffff:32, CharBin/binary >>). 0:192, CharGID:32/little-unsigned-integer, 0:32, 16#ffffffff:32, CharBin/binary >>).
%% @todo Possibly related to 010d. Just send seemingly safe values. %% @todo Possibly related to 010d. Just send seemingly safe values.
send_0111(A, B) -> send_0111(A, B) ->
GID = get(gid), GID = get(gid),
send(<< 16#01110300:32, 0:64, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, send(<< 16#01110300:32, 0:64, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, 0:64,
GID:32/little-unsigned-integer, 0:32, A:32/little-unsigned-integer, B:32/little-unsigned-integer >>). GID:32/little-unsigned-integer, 0:32, A:32/little-unsigned-integer, B:32/little-unsigned-integer >>).
%% @todo Types capability list. %% @todo Types capability list.
send_0113() -> send_0113() ->
{ok, File} = file:read_file("p/typesinfo.bin"), {ok, File} = file:read_file("p/typesinfo.bin"),
GID = get(gid), GID = get(gid),
send(<< 16#01130300:32, 0:64, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, GID:32/little-unsigned-integer, File/binary >>). send(<< 16#01130300:32, 0:64, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, GID:32/little-unsigned-integer, File/binary >>).
%% @doc Update the character's EXP. %% @doc Update the character's EXP.
send_0115(GID, TargetID, LV, EXP, Money) -> send_0115(GID, TargetID, LV, EXP, Money) ->
send(<< 16#01150300:32, 0:64, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, GID:32/little-unsigned-integer, send(<< 16#01150300:32, 0:64, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, GID:32/little-unsigned-integer,
0:32, TargetID:32/little-unsigned-integer, LV:32/little-unsigned-integer, 0:32, 0:32, EXP:32/little-unsigned-integer, 0:32, Money:32/little-unsigned-integer, 16#f5470500:32, 0:96, 0:64, 0:32, TargetID:32/little-unsigned-integer, LV:32/little-unsigned-integer, 0:32, 0:32, EXP:32/little-unsigned-integer, 0:32, Money:32/little-unsigned-integer, 16#f5470500:32, 0:96, 0:64,
@ -1164,14 +1102,12 @@ send_0115(GID, TargetID, LV, EXP, Money) ->
%% @doc Revive player? %% @doc Revive player?
%% @todo Figure out more of it. %% @todo Figure out more of it.
send_0117(HP) -> send_0117(HP) ->
GID = get(gid), GID = get(gid),
send(<< 16#01170300:32, 0:64, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, send(<< 16#01170300:32, 0:64, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, 0:64,
GID:32/little-unsigned-integer, 0:96, HP:32/little-unsigned-integer, 0:32 >>). GID:32/little-unsigned-integer, 0:96, HP:32/little-unsigned-integer, 0:32 >>).
%% @doc Send the zone initialization notification. %% @doc Send the zone initialization notification.
send_0200(ZoneType) -> send_0200(ZoneType) ->
case ZoneType of case ZoneType of
mission -> mission ->
@ -1185,7 +1121,6 @@ send_0200(ZoneType) ->
%% @todo Figure out what the other things are. %% @todo Figure out what the other things are.
%% @todo Handle LID correctly (should be ffffffff for self, apparently). %% @todo Handle LID correctly (should be ffffffff for self, apparently).
send_0201(User) -> send_0201(User) ->
GID = get(gid), GID = get(gid),
CharGID = User#egs_user_model.id, CharGID = User#egs_user_model.id,
@ -1197,12 +1132,10 @@ send_0201(User) ->
GID:32/little-unsigned-integer, 0:64, CharBin/binary, IsGM:8, 0:8, OnlineStatus:8, GameVersion:8, 0:608 >>). GID:32/little-unsigned-integer, 0:64, CharBin/binary, IsGM:8, 0:8, OnlineStatus:8, GameVersion:8, 0:608 >>).
%% @doc Hello packet, always sent on client connection. %% @doc Hello packet, always sent on client connection.
send_0202() -> send_0202() ->
send(<< 16#02020300:32, 0:352 >>). send(<< 16#02020300:32, 0:352 >>).
%% @todo Not sure. Used for unspawning, and more. %% @todo Not sure. Used for unspawning, and more.
send_0204(PlayerGID, PlayerLID, Action) -> send_0204(PlayerGID, PlayerLID, Action) ->
GID = get(gid), GID = get(gid),
send(<< 16#02040300:32, 0:32, 16#00001200:32, PlayerGID:32/little-unsigned-integer, 0:64, send(<< 16#02040300:32, 0:32, 16#00001200:32, PlayerGID:32/little-unsigned-integer, 0:64,
@ -1211,56 +1144,47 @@ send_0204(PlayerGID, PlayerLID, Action) ->
%% @doc Send the map ID to be loaded by the client. %% @doc Send the map ID to be loaded by the client.
%% @todo Last two values are unknown. %% @todo Last two values are unknown.
send_0205(MapType, MapNumber, MapEntry, IsSeasonal) -> send_0205(MapType, MapNumber, MapEntry, IsSeasonal) ->
send(<< 16#02050300:32, 0:288, 16#ffffffff:32, MapType:32/little-unsigned-integer, send(<< 16#02050300:32, 0:288, 16#ffffffff:32, MapType:32/little-unsigned-integer,
MapNumber:32/little-unsigned-integer, MapEntry:32/little-unsigned-integer, 0:56, IsSeasonal >>). MapNumber:32/little-unsigned-integer, MapEntry:32/little-unsigned-integer, 0:56, IsSeasonal >>).
%% @doc Indicate to the client that loading should finish. %% @doc Indicate to the client that loading should finish.
%% @todo Last value seems to be 2 most of the time. Never 0 though. Apparently counters have it at 4. %% @todo Last value seems to be 2 most of the time. Never 0 though. Apparently counters have it at 4.
send_0208() -> send_0208() ->
send(<< (header(16#0208))/binary, 2:32/little-unsigned-integer >>). send(<< (header(16#0208))/binary, 2:32/little-unsigned-integer >>).
%% @todo No idea what this one does. For unknown reasons it uses channel 2. %% @todo No idea what this one does. For unknown reasons it uses channel 2.
send_020c() -> send_020c() ->
send(<< 16#020c020c:32, 16#fffff20c:32, 0:256 >>). send(<< 16#020c020c:32, 16#fffff20c:32, 0:256 >>).
%% @doc Send the quest file to be loaded. %% @doc Send the quest file to be loaded.
%% @todo Probably should try sending the checksum like value (right before the file) and see if it magically fixes anything. %% @todo Probably should try sending the checksum like value (right before the file) and see if it magically fixes anything.
send_020e(Filename) -> send_020e(Filename) ->
{ok, File} = file:read_file(Filename), {ok, File} = file:read_file(Filename),
Size = byte_size(File), Size = byte_size(File),
send(<< 16#020e0300:32, 0:288, Size:32/little-unsigned-integer, 0:32, File/binary, 0:32 >>). send(<< 16#020e0300:32, 0:288, Size:32/little-unsigned-integer, 0:32, File/binary, 0:32 >>).
%% @doc Send the zone file to be loaded. %% @doc Send the zone file to be loaded.
send_020f(Filename, SetID, SeasonID) -> send_020f(Filename, SetID, SeasonID) ->
{ok, File} = file:read_file(Filename), {ok, File} = file:read_file(Filename),
Size = byte_size(File), Size = byte_size(File),
send(<< 16#020f0300:32, 0:288, SetID, SeasonID, 0:16, Size:32/little-unsigned-integer, File/binary >>). send(<< 16#020f0300:32, 0:288, SetID, SeasonID, 0:16, Size:32/little-unsigned-integer, File/binary >>).
%% @doc Send the current UNIX time. %% @doc Send the current UNIX time.
send_0210() -> send_0210() ->
CurrentTime = calendar:datetime_to_gregorian_seconds(calendar:now_to_universal_time(now())) CurrentTime = calendar:datetime_to_gregorian_seconds(calendar:now_to_universal_time(now()))
- calendar:datetime_to_gregorian_seconds({{1970, 1, 1}, {0, 0, 0}}), - calendar:datetime_to_gregorian_seconds({{1970, 1, 1}, {0, 0, 0}}),
send(<< (header(16#0210))/binary, 0:32, CurrentTime:32/little-unsigned-integer >>). send(<< (header(16#0210))/binary, 0:32, CurrentTime:32/little-unsigned-integer >>).
%% @todo No idea what this do. Nor why it's sent twice when loading a counter. %% @todo No idea what this do. Nor why it's sent twice when loading a counter.
send_0215(N) -> send_0215(N) ->
send(<< (header(16#0215))/binary, N:32/little-unsigned-integer >>). send(<< (header(16#0215))/binary, N:32/little-unsigned-integer >>).
%% @todo End of character loading. Just send it. %% @todo End of character loading. Just send it.
send_021b() -> send_021b() ->
send(header(16#021b)). send(header(16#021b)).
%% @doc Send the list of available universes. %% @doc Send the list of available universes.
send_021e() -> send_021e() ->
{ok, << File:1184/bits, _/bits >>} = file:read_file("p/unicube.bin"), {ok, << File:1184/bits, _/bits >>} = file:read_file("p/unicube.bin"),
{ok, Count} = egs_user_model:count(), {ok, Count} = egs_user_model:count(),
@ -1271,7 +1195,6 @@ send_021e() ->
%% @doc Send the current universe name and number. %% @doc Send the current universe name and number.
%% @todo Currently only have universe number 2, named EGS Test. %% @todo Currently only have universe number 2, named EGS Test.
send_0222() -> send_0222() ->
UCS2Name = << << X:8, 0:8 >> || X <- "EGS Test" >>, UCS2Name = << << X:8, 0:8 >> || X <- "EGS Test" >>,
GID = get(gid), GID = get(gid),
@ -1283,13 +1206,11 @@ send_022c(A, B) ->
send(<< (header(16#022c))/binary, A:16/little-unsigned-integer, B:16/little-unsigned-integer >>). send(<< (header(16#022c))/binary, A:16/little-unsigned-integer, B:16/little-unsigned-integer >>).
%% @todo Not sure. Sent when going to or from room. %% @todo Not sure. Sent when going to or from room.
send_0230() -> send_0230() ->
GID = get(gid), GID = get(gid),
send(<< 16#02300300:32, 0:32, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, 0:64 >>). send(<< 16#02300300:32, 0:32, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, 0:64 >>).
%% @todo Figure out what the other things are. %% @todo Figure out what the other things are.
send_0233(Users) -> send_0233(Users) ->
NbUsers = length(Users), NbUsers = length(Users),
case NbUsers of case NbUsers of
@ -1305,7 +1226,6 @@ send_0233(Users) ->
%% @todo God this function is ugly. Use tail recursion! %% @todo God this function is ugly. Use tail recursion!
%% @todo Do it properly without relying on the temporary file. %% @todo Do it properly without relying on the temporary file.
build_0233_contents([]) -> build_0233_contents([]) ->
<< >>; << >>;
build_0233_contents(Users) -> build_0233_contents(Users) ->
@ -1320,12 +1240,10 @@ build_0233_contents(Users) ->
%% @doc Center the camera on the player, if possible. %% @doc Center the camera on the player, if possible.
%% @todo Probably. %% @todo Probably.
send_0236() -> send_0236() ->
send(header(16#0236)). send(header(16#0236)).
%% @doc Send a chat command. Handled differently at v2.0000 and all versions starting somewhere above that. %% @doc Send a chat command. Handled differently at v2.0000 and all versions starting somewhere above that.
send_0304(FromGID, FromName, Modifiers, Message) -> send_0304(FromGID, FromName, Modifiers, Message) ->
case get(version) of case get(version) of
0 -> send(<< 16#03040300:32, 0:288, 16#00001200:32, FromGID:32/little-unsigned-integer, Modifiers:128/bits, Message/bits >>); 0 -> send(<< 16#03040300:32, 0:288, 16#00001200:32, FromGID:32/little-unsigned-integer, Modifiers:128/bits, Message/bits >>);
@ -1334,7 +1252,6 @@ send_0304(FromGID, FromName, Modifiers, Message) ->
%% @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
send_0503(#pos{x=PrevX, y=PrevY, z=PrevZ, dir=_}) -> send_0503(#pos{x=PrevX, y=PrevY, z=PrevZ, dir=_}) ->
{ok, User} = egs_user_model:read(get(gid)), {ok, User} = egs_user_model:read(get(gid)),
#egs_user_model{id=GID, pos=#pos{x=X, y=Y, z=Z, dir=Dir}, area=#psu_area{questid=QuestID, zoneid=ZoneID, mapid=MapID}, entryid=EntryID} = User, #egs_user_model{id=GID, pos=#pos{x=X, y=Y, z=Z, dir=Dir}, area=#psu_area{questid=QuestID, zoneid=ZoneID, mapid=MapID}, entryid=EntryID} = User,
@ -1344,34 +1261,29 @@ send_0503(#pos{x=PrevX, y=PrevY, z=PrevZ, dir=_}) ->
QuestID:32/little-unsigned-integer, ZoneID:32/little-unsigned-integer, MapID:32/little-unsigned-integer, EntryID:32/little-unsigned-integer, 1:32/little-unsigned-integer >>). QuestID:32/little-unsigned-integer, ZoneID:32/little-unsigned-integer, MapID:32/little-unsigned-integer, EntryID:32/little-unsigned-integer, 1:32/little-unsigned-integer >>).
%% @todo Inventory related. No idea what it does. %% @todo Inventory related. No idea what it does.
send_0a05() -> send_0a05() ->
send(header(16#0a05)). send(header(16#0a05)).
%% @todo Inventory related. Figure out everything in this packet and handle it correctly. %% @todo Inventory related. Figure out everything in this packet and handle it correctly.
%% @todo It sends 60 values so it's probably some kind of options for all 60 items in the inventory? %% @todo It sends 60 values so it's probably some kind of options for all 60 items in the inventory?
send_0a06() -> send_0a06() ->
{ok, << _:32, A:96/bits, _:32, B:96/bits, _:32, C:1440/bits, _:32, D/bits >>} = file:read_file("p/packet0a06.bin"), {ok, << _:32, A:96/bits, _:32, B:96/bits, _:32, C:1440/bits, _:32, D/bits >>} = file:read_file("p/packet0a06.bin"),
GID = get(gid), GID = get(gid),
send(<< A/binary, GID:32/little-unsigned-integer, B/binary, GID:32/little-unsigned-integer, C/binary, GID:32/little-unsigned-integer, D/binary >>). send(<< A/binary, GID:32/little-unsigned-integer, B/binary, GID:32/little-unsigned-integer, C/binary, GID:32/little-unsigned-integer, D/binary >>).
%% @todo Inventory. Figure out everything in this packet and handle it correctly. %% @todo Inventory. Figure out everything in this packet and handle it correctly.
send_0a0a() -> send_0a0a() ->
{ok, << _:32, A:224/bits, _:32, B/bits >>} = file:read_file("p/packet0a0a.bin"), {ok, << _:32, A:224/bits, _:32, B/bits >>} = file:read_file("p/packet0a0a.bin"),
GID = get(gid), GID = get(gid),
send(<< A/binary, GID:32/little-unsigned-integer, B/binary >>). send(<< A/binary, GID:32/little-unsigned-integer, B/binary >>).
%% @doc Item description. %% @doc Item description.
send_0a11(ItemID, ItemDesc) -> send_0a11(ItemID, ItemDesc) ->
Size = 1 + length(ItemDesc), Size = 1 + length(ItemDesc),
UCS2Desc = << << X:8, 0:8 >> || X <- ItemDesc >>, UCS2Desc = << << X:8, 0:8 >> || X <- ItemDesc >>,
send(<< (header(16#0a11))/binary, ItemID:32/unsigned-integer, Size:32/little-unsigned-integer, UCS2Desc/binary, 0:16 >>). send(<< (header(16#0a11))/binary, ItemID:32/unsigned-integer, Size:32/little-unsigned-integer, UCS2Desc/binary, 0:16 >>).
%% @doc Init quest. %% @doc Init quest.
send_0c00(QuestID) -> send_0c00(QuestID) ->
send(<< (header(16#0c00))/binary, QuestID:32/little-unsigned-integer, send(<< (header(16#0c00))/binary, QuestID:32/little-unsigned-integer,
16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32,
@ -1380,30 +1292,25 @@ send_0c00(QuestID) ->
16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32 >>). 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32 >>).
%% @todo Figure out last 4 bytes! %% @todo Figure out last 4 bytes!
send_0c02() -> send_0c02() ->
send(<< (header(16#0c02))/binary, 0:32 >>). send(<< (header(16#0c02))/binary, 0:32 >>).
%% @doc Send the huge pack of quest files available in the counter. %% @doc Send the huge pack of quest files available in the counter.
send_0c06(Filename) -> send_0c06(Filename) ->
{ok, << File/bits >>} = file:read_file(Filename), {ok, << File/bits >>} = file:read_file(Filename),
send(<< 16#0c060300:32, 0:288, 1:32/little-unsigned-integer, File/binary >>). send(<< 16#0c060300:32, 0:288, 1:32/little-unsigned-integer, File/binary >>).
%% @doc Reply whether the player is allowed to use the transport option. %% @doc Reply whether the player is allowed to use the transport option.
%% Use true for allowing it, and false otherwise. %% Use true for allowing it, and false otherwise.
send_0c08(Response) -> send_0c08(Response) ->
Value = if Response =:= true -> 0; true -> 1 end, Value = if Response =:= true -> 0; true -> 1 end,
send(<< (header(16#0c08))/binary, Value:32 >>). send(<< (header(16#0c08))/binary, Value:32 >>).
%% @doc Send the trial start notification. %% @doc Send the trial start notification.
send_0c09() -> send_0c09() ->
send(<< (header(16#0c09))/binary, 0:64 >>). send(<< (header(16#0c09))/binary, 0:64 >>).
%% @doc Send the counter's mission options (0 = invisible, 2 = disabled, 3 = available). %% @doc Send the counter's mission options (0 = invisible, 2 = disabled, 3 = available).
send_0c10(Options) -> send_0c10(Options) ->
GID = get(gid), GID = get(gid),
send(<< 16#0c100300:32, 0:32, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, send(<< 16#0c100300:32, 0:32, 16#00011300:32, GID:32/little-unsigned-integer, 0:64,
@ -1412,7 +1319,6 @@ send_0c10(Options) ->
%% @doc Send the data for the selected character. %% @doc Send the data for the selected character.
%% @todo The large chunk of 0s can have some values set... but what are they used for? %% @todo The large chunk of 0s can have some values set... but what are they used for?
%% @todo The values after the Char variable are the flags. Probably use bits to define what flag is and isn't set. Handle correctly. %% @todo The values after the Char variable are the flags. Probably use bits to define what flag is and isn't set. Handle correctly.
send_0d01(User) -> send_0d01(User) ->
CharBin = psu_characters:character_tuple_to_binary(User#egs_user_model.character), CharBin = psu_characters:character_tuple_to_binary(User#egs_user_model.character),
OptionsBin = psu_characters:options_tuple_to_binary((User#egs_user_model.character)#characters.options), OptionsBin = psu_characters:options_tuple_to_binary((User#egs_user_model.character)#characters.options),
@ -1424,7 +1330,6 @@ send_0d01(User) ->
%% @doc Send the character list for selection. %% @doc Send the character list for selection.
%% @todo There's a few odd values blanked, also the last known location apparently. %% @todo There's a few odd values blanked, also the last known location apparently.
send_0d03(Data0, Data1, Data2, Data3) -> send_0d03(Data0, Data1, Data2, Data3) ->
[{status, Status0}, {char, Char0}|_] = Data0, [{status, Status0}, {char, Char0}|_] = Data0,
[{status, Status1}, {char, Char1}|_] = Data1, [{status, Status1}, {char, Char1}|_] = Data1,
@ -1439,7 +1344,6 @@ send_0d03(Data0, Data1, Data2, Data3) ->
%% @doc Send the character flags list. This is the whole list of available values, not the character's. %% @doc Send the character flags list. This is the whole list of available values, not the character's.
%% Sent without fragmentation on official for unknown reasons. Do the same here. %% Sent without fragmentation on official for unknown reasons. Do the same here.
send_0d05() -> send_0d05() ->
{ok, Flags} = file:read_file("p/flags.bin"), {ok, Flags} = file:read_file("p/flags.bin"),
GID = get(gid), GID = get(gid),
@ -1464,7 +1368,6 @@ send_1004(GID, NPCid, Name, Level, PartyPos) ->
16#12000000:32, 16#24e41f08:32, 0:128, 16#ffffffff:32, 0:64, 16#01000000:32, 16#01000000:32, 0:608 >>). 16#12000000:32, 16#24e41f08:32, 0:128, 16#ffffffff:32, 0:64, 16#01000000:32, 16#01000000:32, 0:608 >>).
%% @todo Figure out what the packet is. %% @todo Figure out what the packet is.
send_1005(Name) -> send_1005(Name) ->
{ok, File} = file:read_file("p/packet1005.bin"), {ok, File} = file:read_file("p/packet1005.bin"),
<< _:352, Before:160/bits, _:608, After/bits >> = File, << _:352, Before:160/bits, _:608, After/bits >> = File,
@ -1474,12 +1377,10 @@ send_1005(Name) ->
%% @doc Party-related command probably controlling the party state. %% @doc Party-related command probably controlling the party state.
%% Value 11 aborts the mission. %% Value 11 aborts the mission.
%% @todo Figure out what the packet is. %% @todo Figure out what the packet is.
send_1006(N) -> send_1006(N) ->
send(<< (header(16#1006))/binary, N:32/little-unsigned-integer >>). send(<< (header(16#1006))/binary, N:32/little-unsigned-integer >>).
%% @doc Send the player's current location. %% @doc Send the player's current location.
send_100e(QuestID, ZoneID, MapID, Location, CounterID) -> send_100e(QuestID, ZoneID, MapID, Location, CounterID) ->
UCS2Location = << << X:8, 0:8 >> || X <- Location >>, UCS2Location = << << X:8, 0:8 >> || X <- Location >>,
Packet = << (header(16#100e))/binary, 1:32/little-unsigned-integer, MapID:16/little-unsigned-integer, Packet = << (header(16#100e))/binary, 1:32/little-unsigned-integer, MapID:16/little-unsigned-integer,
@ -1495,7 +1396,6 @@ send_100e(QuestID, ZoneID, MapID, Location, CounterID) ->
%% @doc Send the mission's quest file when starting a new mission. %% @doc Send the mission's quest file when starting a new mission.
%% @todo Handle correctly. 0:32 is actually a missing value. Value before that is unknown too. %% @todo Handle correctly. 0:32 is actually a missing value. Value before that is unknown too.
send_1015(QuestID) -> send_1015(QuestID) ->
[{type, _}, {file, QuestFile}|_] = proplists:get_value(QuestID, ?QUESTS), [{type, _}, {file, QuestFile}|_] = proplists:get_value(QuestID, ?QUESTS),
{ok, File} = file:read_file(QuestFile), {ok, File} = file:read_file(QuestFile),
@ -1503,71 +1403,58 @@ send_1015(QuestID) ->
send(<< (header(16#1015))/binary, QuestID:32/little-unsigned-integer, 16#01010000:32, 0:32, Size:32/little-unsigned-integer, File/binary >>). send(<< (header(16#1015))/binary, QuestID:32/little-unsigned-integer, 16#01010000:32, 0:32, Size:32/little-unsigned-integer, File/binary >>).
%% @todo Totally unknown. %% @todo Totally unknown.
send_1020() -> send_1020() ->
send(header(16#1020)). send(header(16#1020)).
%% @doc Update HP in the party members information on the left. %% @doc Update HP in the party members information on the left.
%% @todo Figure out more of it. %% @todo Figure out more of it.
send_1022(HP) -> send_1022(HP) ->
GID = get(gid), GID = get(gid),
send(<< 16#10220300:32, 0:160, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, HP:32/little-unsigned-integer, 0:32 >>). send(<< 16#10220300:32, 0:160, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, HP:32/little-unsigned-integer, 0:32 >>).
%% @todo Boss related command. %% @todo Boss related command.
send_110e(Data) -> send_110e(Data) ->
send(<< (header(16#110e))/binary, Data/binary, 0:32, 5:16/little-unsigned-integer, 12:16/little-unsigned-integer, 0:32, 260:32/little-unsigned-integer >>). send(<< (header(16#110e))/binary, Data/binary, 0:32, 5:16/little-unsigned-integer, 12:16/little-unsigned-integer, 0:32, 260:32/little-unsigned-integer >>).
%% @todo Boss related command. %% @todo Boss related command.
send_1113(Data) -> send_1113(Data) ->
send(<< (header(16#1113))/binary, Data/binary >>). send(<< (header(16#1113))/binary, Data/binary >>).
%% @todo Figure out what this packet does. Sane values for counter and missions for now. %% @todo Figure out what this packet does. Sane values for counter and missions for now.
send_1202() -> send_1202() ->
send(<< (header(16#1202))/binary, 0:32, 16#10000000:32, 0:64, 16#14000000:32, 0:32 >>). send(<< (header(16#1202))/binary, 0:32, 16#10000000:32, 0:64, 16#14000000:32, 0:32 >>).
%% @todo Figure out what this packet does. Seems it's the same values all the time. %% @todo Figure out what this packet does. Seems it's the same values all the time.
send_1204() -> send_1204() ->
send(<< (header(16#1204))/binary, 0:32, 16#20000000:32, 0:256 >>). send(<< (header(16#1204))/binary, 0:32, 16#20000000:32, 0:256 >>).
%% @doc Object events response? %% @doc Object events response?
%% @todo Not sure what Value does exactly. It's either 0 or 1. %% @todo Not sure what Value does exactly. It's either 0 or 1.
send_1205(EventID, BlockID, Value) -> send_1205(EventID, BlockID, Value) ->
send(<< (header(16#1205))/binary, EventID, BlockID, 0:16, Value, 0:24 >>). send(<< (header(16#1205))/binary, EventID, BlockID, 0:16, Value, 0:24 >>).
%% @todo Figure out what this packet does. Sane values for counter and missions for now. %% @todo Figure out what this packet does. Sane values for counter and missions for now.
send_1206() -> send_1206() ->
send(<< (header(16#1206))/binary, 0:32, 16#80020000:32, 0:5120 >>). send(<< (header(16#1206))/binary, 0:32, 16#80020000:32, 0:5120 >>).
%% @todo Figure out what this packet does. Sane values for counter and missions for now. %% @todo Figure out what this packet does. Sane values for counter and missions for now.
send_1207() -> send_1207() ->
Chunk = << 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 0:224, 16#0000ffff:32, 16#ff000000:32, 16#64000a00:32 >>, Chunk = << 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 0:224, 16#0000ffff:32, 16#ff000000:32, 16#64000a00:32 >>,
send(<< (header(16#1207))/binary, Chunk/binary, Chunk/binary, Chunk/binary, Chunk/binary, Chunk/binary, Chunk/binary >>). send(<< (header(16#1207))/binary, Chunk/binary, Chunk/binary, Chunk/binary, Chunk/binary, Chunk/binary, Chunk/binary >>).
%% @todo Object interaction? Figure out. C probably the interaction type. %% @todo Object interaction? Figure out. C probably the interaction type.
send_1211(A, B, C, D) -> send_1211(A, B, C, D) ->
send(<< (header(16#1211))/binary, A:32/little-unsigned-integer, B:32/little-unsigned-integer, C:32/little-unsigned-integer, D:32/little-unsigned-integer >>). send(<< (header(16#1211))/binary, A:32/little-unsigned-integer, B:32/little-unsigned-integer, C:32/little-unsigned-integer, D:32/little-unsigned-integer >>).
%% @doc Make the client load the quest previously sent. %% @doc Make the client load the quest previously sent.
send_1212() -> send_1212() ->
send(<< (header(16#1212))/binary, 0:19200 >>). send(<< (header(16#1212))/binary, 0:19200 >>).
%% @todo Not sure. Related to keys. %% @todo Not sure. Related to keys.
send_1213(A, B) -> send_1213(A, B) ->
send(<< (header(16#1213))/binary, A:32/little-unsigned-integer, B:32/little-unsigned-integer >>). send(<< (header(16#1213))/binary, A:32/little-unsigned-integer, B:32/little-unsigned-integer >>).
%% @todo Related to boss gates. %% @todo Related to boss gates.
send_1215(A, B) -> send_1215(A, B) ->
send(<< (header(16#1215))/binary, A:32/little-unsigned-integer, 0:16, B:16/little-unsigned-integer >>). send(<< (header(16#1215))/binary, A:32/little-unsigned-integer, 0:16, B:16/little-unsigned-integer >>).
@ -1578,7 +1465,6 @@ send_1216(Value) ->
%% @doc Send the player's partner card. %% @doc Send the player's partner card.
%% @todo Find out the remaining values. %% @todo Find out the remaining values.
send_1500(User) -> send_1500(User) ->
#characters{slot=Slot, name=Name, race=Race, gender=Gender, class=Class} = User#egs_user_model.character, #characters{slot=Slot, name=Name, race=Race, gender=Gender, class=Class} = User#egs_user_model.character,
RaceBin = psu_characters:race_atom_to_binary(Race), RaceBin = psu_characters:race_atom_to_binary(Race),
@ -1587,12 +1473,10 @@ send_1500(User) ->
send(<< (header(16#1500))/binary, Name/binary, RaceBin:8, GenderBin:8, ClassBin:8, 0:3112, 16#010401:24, Slot:8, 0:64 >>). send(<< (header(16#1500))/binary, Name/binary, RaceBin:8, GenderBin:8, ClassBin:8, 0:3112, 16#010401:24, Slot:8, 0:64 >>).
%% @todo Send an empty partner card list. %% @todo Send an empty partner card list.
send_1501() -> send_1501() ->
send(<< (header(16#1501))/binary, 0:32 >>). send(<< (header(16#1501))/binary, 0:32 >>).
%% @todo Send an empty blacklist. %% @todo Send an empty blacklist.
send_1512() -> send_1512() ->
send(<< (header(16#1512))/binary, 0:46080 >>). send(<< (header(16#1512))/binary, 0:46080 >>).
@ -1610,59 +1494,50 @@ send_1602() ->
%% @doc Party information. %% @doc Party information.
%% @todo Handle existing parties. %% @todo Handle existing parties.
send_1706(CharName) -> send_1706(CharName) ->
send(<< (header(16#1706))/binary, 16#00000300:32, 16#d5c0faff:32, 0:64, CharName/binary, send(<< (header(16#1706))/binary, 16#00000300:32, 16#d5c0faff:32, 0:64, CharName/binary,
16#78000000:32, 16#01010000:32, 0:1536, 16#0100c800:32, 16#0601010a:32, 16#ffffffff:32, 0:32 >>). 16#78000000:32, 16#01010000:32, 0:1536, 16#0100c800:32, 16#0601010a:32, 16#ffffffff:32, 0:32 >>).
%% @doc Party settings. Item distribution is random for now. %% @doc Party settings. Item distribution is random for now.
%% @todo Handle correctly. %% @todo Handle correctly.
send_170a() -> send_170a() ->
send(<< (header(16#170a))/binary, 16#01010c08:32 >>). send(<< (header(16#170a))/binary, 16#01010c08:32 >>).
%% @todo Find what the heck this packet is. %% @todo Find what the heck this packet is.
send_170c() -> send_170c() ->
{ok, File} = file:read_file("p/packet170c.bin"), {ok, File} = file:read_file("p/packet170c.bin"),
send(<< (header(16#170c))/binary, File/binary >>). send(<< (header(16#170c))/binary, File/binary >>).
%% @doc Send the background to use for the counter. %% @doc Send the background to use for the counter.
%% @todo Background has more info past the first byte. %% @todo Background has more info past the first byte.
send_1711(Background) -> send_1711(Background) ->
send(<< (header(16#1711))/binary, Background:32/little-unsigned-integer >>). send(<< (header(16#1711))/binary, Background:32/little-unsigned-integer >>).
%% @doc Unknown dialog-related handler. %% @doc Unknown dialog-related handler.
%% @todo Everything! %% @todo Everything!
send_1a02(A, B, C, D, E) -> send_1a02(A, B, C, D, E) ->
send(<< (header(16#1a02))/binary, A:32/little-unsigned-integer, B:16/little-unsigned-integer, send(<< (header(16#1a02))/binary, A:32/little-unsigned-integer, B:16/little-unsigned-integer,
C:16/little-unsigned-integer, D:16/little-unsigned-integer, E:16/little-unsigned-integer >>). C:16/little-unsigned-integer, D:16/little-unsigned-integer, E:16/little-unsigned-integer >>).
%% @doc Lumilass handler. Possibly more. %% @doc Lumilass handler. Possibly more.
%% @todo Figure out how Lumilass work exactly. The 4 bytes before the file may vary. %% @todo Figure out how Lumilass work exactly. The 4 bytes before the file may vary.
send_1a03() -> send_1a03() ->
{ok, File} = file:read_file("p/lumilassA.bin"), {ok, File} = file:read_file("p/lumilassA.bin"),
send(<< (header(16#1a03))/binary, 0:32, File/binary >>). send(<< (header(16#1a03))/binary, 0:32, File/binary >>).
%% @doc PP cube handler. %% @doc PP cube handler.
%% @todo The 4 bytes before the file may vary. Everything past that is the same. Figure things out. %% @todo The 4 bytes before the file may vary. Everything past that is the same. Figure things out.
send_1a04() -> send_1a04() ->
{ok, File} = file:read_file("p/ppcube.bin"), {ok, File} = file:read_file("p/ppcube.bin"),
send(<< (header(16#1a04))/binary, 0:32, File/binary >>). send(<< (header(16#1a04))/binary, 0:32, File/binary >>).
%% @doc Types menu handler. %% @doc Types menu handler.
%% @todo Handle correctly. %% @todo Handle correctly.
send_1a07() -> send_1a07() ->
send(<< (header(16#1a07))/binary, 16#085b5d0a:32, 16#3a200000:32, 0:32, send(<< (header(16#1a07))/binary, 16#085b5d0a:32, 16#3a200000:32, 0:32,
16#01010101:32, 16#01010101:32, 16#01010101:32, 16#01010101:32 >>). 16#01010101:32, 16#01010101:32, 16#01010101:32, 16#01010101:32 >>).
%% @doc Log message to the console. %% @doc Log message to the console.
log(Message) -> log(Message) ->
io:format("game (~p): ~s~n", [get(gid), Message]). io:format("game (~p): ~s~n", [get(gid), Message]).