%% @author Loïc Hoguin %% @copyright 2010 Loïc Hoguin. %% @doc Handle game clients. %% %% This file is part of EGS. %% %% EGS is free software: you can redistribute it and/or modify %% it under the terms of the GNU Affero General Public License as %% published by the Free Software Foundation, either version 3 of the %% License, or (at your option) any later version. %% %% EGS is distributed in the hope that it will be useful, %% but WITHOUT ANY WARRANTY; without even the implied warranty of %% MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the %% GNU Affero General Public License for more details. %% %% You should have received a copy of the GNU Affero General Public License %% along with EGS. If not, see . -module(psu_game). -export([start_link/1, cleanup/1]). %% External. -export([listen/2, accept/2, process_init/2, process/0, char_select/0, loop/1]). %% Internal. -include("include/records.hrl"). -include("include/maps.hrl"). -include("include/missions.hrl"). -include("include/psu/items.hrl"). -include("include/psu/npc.hrl"). -define(OPTIONS, [binary, {active, false}, {reuseaddr, true}, {certfile, "priv/ssl/servercert.pem"}, {keyfile, "priv/ssl/serverkey.pem"}, {password, "alpha"}]). %% @spec start_link(Port) -> {ok,Pid::pid()} %% @doc Start the game server. start_link(Port) -> {ok, MPid} = egs_exit_mon:start_link({?MODULE, cleanup}), LPid = spawn(?MODULE, listen, [Port, MPid]), {ok, LPid}. %% @spec cleanup(Pid) -> ok %% @doc Cleanup the data associated with the failing process. %% @todo Cleanup the instance process if there's nobody in it anymore. %% @todo Leave party instead of stopping it. cleanup(Pid) -> case egs_user_model:read({pid, Pid}) of {ok, User} -> case User#egs_user_model.partypid of undefined -> ignore; PartyPid -> {ok, NPCList} = psu_party:get_npc(PartyPid), [egs_user_model:delete(NPCGID) || {_Spot, NPCGID} <- NPCList], psu_party:stop(PartyPid) end, egs_user_model:delete(User#egs_user_model.id), {ok, List} = egs_user_model:select({neighbors, User}), lists:foreach(fun(Other) -> Other#egs_user_model.pid ! {psu_player_unspawn, User} end, List), io:format("game (~p): quit~n", [User#egs_user_model.id]); {error, _Reason} -> ignore end. %% @doc Listen for connections. listen(Port, MPid) -> error_logger:info_report(io_lib:format("psu_game listening on port ~b", [Port])), {ok, LSocket} = ssl:listen(Port, ?OPTIONS), ?MODULE:accept(LSocket, MPid). %% @doc Accept connections. accept(LSocket, MPid) -> case ssl:transport_accept(LSocket, 5000) of {ok, CSocket} -> case ssl:ssl_accept(CSocket, 5000) of ok -> Pid = spawn(?MODULE, process_init, [CSocket, MPid]), ssl:controlling_process(CSocket, Pid); {error, _Reason} -> reload end; _ -> reload end, ?MODULE:accept(LSocket, MPid). %% @doc Initialize the client process by saving the socket to the process dictionary. process_init(CSocket, MPid) -> link(MPid), put(socket, CSocket), psu_proto:send_0202(CSocket, 0), timer:send_interval(5000, {psu_keepalive}), process(). %% @doc Process the new connections. %% Send an hello packet, authenticate the user and send him to character select. process() -> case psu_proto:packet_recv(get(socket), 5000) of {ok, Orig} -> case psu_proto:parse(Orig) of ignore -> ?MODULE:process(); Event -> process_event(Event) end; {error, timeout} -> reload, ?MODULE:process(); {error, closed} -> closed end. %% @todo Check the client version info here too. Not just in psu_login. process_event({system_client_version_info, _Language, _Platform, _Version}) -> ?MODULE:process(); process_event({system_key_auth_request, AuthGID, AuthKey}) -> CSocket = get(socket), case egs_user_model:read(AuthGID) of {error, badarg} -> log("can't find user, closing"), ssl:close(CSocket); {ok, User} -> case User#egs_user_model.state of {wait_for_authentication, AuthKey} -> put(gid, AuthGID), log("auth success"), LID = 1 + mnesia:dirty_update_counter(counters, lobby, 1) rem 1023, Time = calendar:datetime_to_gregorian_seconds(calendar:universal_time()), NewUser = #egs_user_model{id=AuthGID, pid=self(), socket=CSocket, state=authenticated, time=Time, folder=User#egs_user_model.folder, lid=LID}, egs_user_model:write(NewUser), psu_proto:send_0d05(NewUser), ?MODULE:char_select(); _ -> log("quit, auth failed"), ssl:close(CSocket) end end; process_event({command, Command, Channel, _Data}) -> log("process_event: dismissed command ~4.16.0b channel ~b", [Command, Channel]), ?MODULE:process(). %% @doc Character selection screen loop. char_select() -> case psu_proto:packet_recv(get(socket), 5000) of {ok, Orig} -> case psu_proto:parse(Orig) of ignore -> ?MODULE:char_select(); Event -> char_select_event(Event) end; {error, timeout} -> psu_proto:send_keepalive(get(socket)), reload, ?MODULE:char_select(); {error, closed} -> closed %% exit end. %% @todo Reenable appearance validation whenever things go live. char_select_event({char_select_create, Slot, CharBin}) -> log("character creation ~b", [Slot]), %% check for valid character appearance %~ << _Name:512, RaceID:8, GenderID:8, _TypeID:8, AppearanceBin:776/bits, _/bits >> = CharBin, %~ Race = proplists:get_value(RaceID, [{0, human}, {1, newman}, {2, cast}, {3, beast}]), %~ Gender = proplists:get_value(GenderID, [{0, male}, {1, female}]), %~ Appearance = psu_appearance:binary_to_tuple(Race, AppearanceBin), %~ psu_appearance:validate_char_create(Race, Gender, Appearance), %% end of check, continue doing it wrong past that point for now {ok, User} = egs_user_model:read(get(gid)), _ = file:make_dir(io_lib:format("save/~s", [User#egs_user_model.folder])), file:write_file(io_lib:format("save/~s/~b-character", [User#egs_user_model.folder, Slot]), CharBin), file:write_file(io_lib:format("save/~s/~b-character.options", [User#egs_user_model.folder, Slot]), << 0:128, 4, 0:56 >>), % default 0 to everything except brightness 4 ?MODULE:char_select(); char_select_event({char_select_enter, Slot, _BackToPreviousField}) -> log("selected character ~b", [Slot]), char_select_load(Slot); char_select_event(char_select_request) -> {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)), ?MODULE:char_select(); char_select_event({command, Command, Channel, _Data}) -> log("char_select_event: dismissed command ~4.16.0b channel ~b", [Command, Channel]), ?MODULE:char_select(). %% @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. char_select_load(Number) -> {ok, OldUser} = egs_user_model:read(get(gid)), [{status, 1}, {char, CharBin}, {options, OptionsBin}] = data_load(OldUser#egs_user_model.folder, Number), << Name:512/bits, RaceBin:8, GenderBin:8, ClassBin:8, AppearanceBin:776/bits, _/bits >> = CharBin, psu_characters:validate_name(Name), % TODO: don't validate name when loading character, do it at creation Race = psu_characters:race_binary_to_atom(RaceBin), Gender = psu_characters:gender_binary_to_atom(GenderBin), Class = psu_characters:class_binary_to_atom(ClassBin), Appearance = psu_appearance:binary_to_tuple(Race, AppearanceBin), Options = psu_characters:options_binary_to_tuple(OptionsBin), Character = #characters{slot=Number, name=Name, race=Race, gender=Gender, class=Class, appearance=Appearance, options=Options}, % TODO: temporary set the slot here, won't be needed later User = OldUser#egs_user_model{state=online, character=Character, area=#psu_area{questid=undefined, zoneid=undefined, mapid=undefined}, prev_area={psu_area, 0, 0, 0}, prev_entryid=0, pos=#pos{x=0.0, y=0.0, z=0.0, dir=0.0}, setid=0}, egs_user_model:write(User), char_load(User), event({area_change, 1100000, 0, 4, 5}), ssl:setopts(get(socket), [{active, true}]), ?MODULE:loop(<< >>). %% @doc Load the given character's data. data_load(Folder, Number) -> Filename = io_lib:format("save/~s/~b-character", [Folder, Number]), case file:read_file(Filename) of {ok, Char} -> {ok, Options} = file:read_file(io_lib:format("~s.options", [Filename])), [{status, 1}, {char, Char}, {options, Options}]; {error, _} -> [{status, 0}, {char, << 0:2208 >>}] end. %% @doc Load and send the character information to the client. %% @todo Should wait for the 021c reply before doing area_change. char_load(User) -> send_0d01(User), % 0246 send_0a0a(), send_1006(5, 0), send_1005((User#egs_user_model.character)#characters.name), send_1006(12, 0), send_0210(), send_0222(), send_1500(User), send_1501(), send_1512(), % 0303 send_1602(), send_021b(). %% @doc Return the current season information. area_get_season(QuestID) -> {{_, Month, Day}, _} = calendar:universal_time(), [IsSeasonal, SeasonID, SeasonQuestIDs] = if Month =:= 1, Day =< 14 -> ?SEASON_NEWYEAR; Month =:= 1, Day >= 25 -> ?SEASON_WINTER; Month =:= 2, Day =< 7 -> ?SEASON_WINTER; Month =:= 2, Day >= 14 -> ?SEASON_VALENTINE; Month =:= 3, Day =< 6 -> ?SEASON_VALENTINE; Month =:= 3, Day >= 14 -> ?SEASON_WHITEDAY; Month =:= 4, Day =< 3 -> ?SEASON_WHITEDAY; Month =:= 4, Day >= 4, Day =< 24 -> ?SEASON_EASTER; Month =:= 4, Day >= 25 -> ?SEASON_SPRING; Month =:= 5, Day =< 8 -> ?SEASON_SPRING; Month =:= 5, Day >= 17, Day =< 30 -> ?SEASON_WEDDING; Month =:= 6, Day >= 3, Day =< 16 -> ?SEASON_PARUMUNIF; Month =:= 6, Day >= 23 -> ?SEASON_SONIC; Month =:= 7, Day =< 13 -> ?SEASON_SONIC; Month =:= 7, Day >= 18 -> ?SEASON_HOLYLIGHT; Month =:= 8, Day =< 21 -> ?SEASON_FIREWORKS; Month =:= 8, Day >= 28 -> ?SEASON_NATIVE; Month =:= 9, Day =< 10 -> ?SEASON_NATIVE; Month =:= 9, Day >= 24 -> ?SEASON_AUTUMN; Month =:= 10, Day =< 7 -> ?SEASON_AUTUMN; Month =:= 10, Day >= 15, Day =< 28 -> ?SEASON_PARTY; Month =:= 10, Day >= 31 -> ?SEASON_HALLOWEEN; Month =:= 11, Day =< 20 -> ?SEASON_HALLOWEEN; Month =:= 12, Day >= 11 -> ?SEASON_CHRISTMAS; true -> ?SEASON_NONE end, if IsSeasonal =:= 1 -> case lists:member(QuestID, SeasonQuestIDs) of true -> [{status, IsSeasonal}, {season, SeasonID}]; false -> [{status, 0}, {season, 255}] end; true -> [{status, 0}, {season, 255}] end. %% @doc Load the given map as a standard lobby. area_load(QuestID, ZoneID, MapID, EntryID) -> {ok, OldUser} = egs_user_model:read(get(gid)), [{type, AreaType}, {file, QuestFile}|MissionInfo] = proplists:get_value(QuestID, ?QUESTS, [{type, undefined}, {file, undefined}]), [IsStart, RealZoneID, RealMapID, RealEntryID, NbSetsInQuest] = case AreaType of mission -> if ZoneID =:= 65535 -> [{start, [TmpZoneID, TmpMapID, TmpEntryID]}, {sets, TmpNbSets}] = MissionInfo, [true, TmpZoneID, TmpMapID, TmpEntryID, TmpNbSets]; true -> [false, ZoneID, MapID, EntryID, ignored] end; myroom -> if ZoneID =:= 0 -> [false, 0, 423, EntryID, ignored]; true -> [false, ZoneID, MapID, EntryID, ignored] end; _ -> [false, ZoneID, MapID, EntryID, ignored] end, [{file, ZoneFile}|ZoneSetInfo] = proplists:get_value([QuestID, RealZoneID], ?ZONES, [{file, undefined}]), NbSetsInZone = case ZoneSetInfo of [] -> 1; [{sets, TmpNbSetsInZone}] -> TmpNbSetsInZone end, if AreaType =:= myroom -> AreaName = "Your Room"; true -> [{name, AreaName}] = proplists:get_value([QuestID, RealMapID], ?MAPS, [{name, "dammy"}]) end, {InstancePid, SetID} = if IsStart =:= true -> % initialize the mission Zones = proplists:get_value(QuestID, ?MISSIONS), {ok, RetPid} = psu_instance:start_link(Zones), RetSetID = crypto:rand_uniform(0, NbSetsInQuest), {RetPid, RetSetID}; true -> {OldUser#egs_user_model.instancepid, OldUser#egs_user_model.setid} end, User = OldUser#egs_user_model{instancepid=InstancePid, areatype=AreaType, area={psu_area, QuestID, RealZoneID, RealMapID}, entryid=RealEntryID}, egs_user_model:write(User), RealSetID = if SetID > NbSetsInZone - 1 -> NbSetsInZone - 1; true -> SetID end, area_load(AreaType, IsStart, RealSetID, OldUser, User, QuestFile, ZoneFile, AreaName). area_load(AreaType, IsStart, SetID, OldUser, User, QuestFile, ZoneFile, AreaName) -> #psu_area{questid=OldQuestID, zoneid=OldZoneID} = OldUser#egs_user_model.area, #psu_area{questid=QuestID, zoneid=ZoneID, mapid=_MapID} = User#egs_user_model.area, QuestChange = if OldQuestID /= QuestID, QuestFile /= undefined -> true; true -> false end, if ZoneFile =:= undefined -> ZoneChange = false; true -> ZoneChange = if OldQuestID =:= QuestID, OldZoneID =:= ZoneID -> false; true -> true end end, [{status, IsSeasonal}, {season, SeasonID}] = area_get_season(QuestID), % broadcast spawn and unspawn to other people {ok, UnspawnList} = egs_user_model:select({neighbors, OldUser}), {ok, SpawnList} = egs_user_model:select({neighbors, User}), lists:foreach(fun(Other) -> Other#egs_user_model.pid ! {psu_player_unspawn, User} end, UnspawnList), if AreaType =:= lobby -> lists:foreach(fun(Other) -> Other#egs_user_model.pid ! {psu_player_spawn, User} end, SpawnList); true -> ignore end, % load area if QuestChange =:= true -> % reload the character if entering or leaving the room quest if OldQuestID =:= 1120000; QuestID =:= 1120000 -> char_load(User); true -> ignore end, % load new quest psu_proto:send_0c00(User), psu_proto:send_020e(User, QuestFile); true -> ignore end, if IsStart =:= true -> psu_proto:send_0215(User#egs_user_model{lid=0}, 16#ffffffff); true -> ignore end, if ZoneChange =:= true -> % load new zone psu_proto:send_0a05(User#egs_user_model{lid=0}), if AreaType =:= lobby -> send_0111(6, 0); true -> ignore end, psu_proto:send_010d(User, User#egs_user_model{lid=0}), send_0200(AreaType), send_020f(ZoneFile, SetID, SeasonID); true -> ignore end, psu_proto:send_0205(User#egs_user_model{lid=0}, IsSeasonal), send_100e(QuestID, ZoneID, (User#egs_user_model.area)#psu_area.mapid, AreaName, 16#ffffffff), if AreaType =:= mission -> psu_proto:send_0215(User#egs_user_model{lid=0}, 0), if IsStart =:= true -> psu_proto:send_0215(User#egs_user_model{lid=0}, 0), send_0c09(); true -> ignore end; true -> psu_proto:send_020c(User) end, if ZoneChange =:= true -> case AreaType of myroom -> send_1332(), send_1202(), send_1204(), send_1206(); mission -> send_1202(), send_1204(), send_1206(), send_1207(); _ -> ignore end; true -> ignore end, if ZoneChange =:= true, AreaType =/= spaceport -> send_1212(); true -> ignore end, if AreaType =:= myroom -> send_1309(); true -> ignore end, psu_proto:send_0201(User, User#egs_user_model{lid=0}), if ZoneChange =:= true -> send_0a06(); true -> ignore end, send_0233(SpawnList), case User#egs_user_model.partypid of undefined -> ignore; _ -> send_022c(0, 16#12) end, send_0208(), send_0236(), if User#egs_user_model.partypid =/= undefined, AreaType =:= mission -> {ok, NPCList} = psu_party:get_npc(User#egs_user_model.partypid), npc_load(User, NPCList); true -> ok end. %% @todo Don't change the NPC info unless you are the leader! npc_load(_Leader, []) -> ok; npc_load(Leader, [{PartyPos, NPCGID}|NPCList]) -> {ok, OldNPCUser} = egs_user_model:read(NPCGID), #egs_user_model{instancepid=InstancePid, area=Area, entryid=EntryID, pos=Pos} = Leader, NPCUser = OldNPCUser#egs_user_model{lid=PartyPos, instancepid=InstancePid, areatype=mission, area=Area, entryid=EntryID, pos=Pos}, %% @todo This one on mission end/abort? %~ OldNPCUser#egs_user_model{lid=PartyPos, instancepid=undefined, areatype=AreaType, area={psu_area, 0, 0, 0}, entryid=0, pos={pos, 0.0, 0.0, 0.0, 0}} egs_user_model:write(NPCUser), psu_proto:send_010d(Leader, NPCUser), psu_proto:send_0201(Leader, NPCUser), psu_proto:send_0215(Leader, 0), send_0a04(NPCUser#egs_user_model.id), send_1004(npc_mission, NPCUser, PartyPos), send_100f((NPCUser#egs_user_model.character)#characters.npcid, PartyPos), send_1601(PartyPos), send_1016(PartyPos), npc_load(Leader, NPCList). %% @doc Game's main loop. %% @todo We probably don't want to send a keepalive packet unnecessarily. loop(SoFar) -> receive {psu_broadcast, Orig} -> << A:64/bits, _:32, B:96/bits, _:64, C/bits >> = Orig, GID = get(gid), send(<< A/binary, 16#00011300:32, B/binary, 16#00011300:32, GID:32/little-unsigned-integer, C/binary >>), ?MODULE:loop(SoFar); {psu_chat, ChatTypeID, ChatGID, ChatName, ChatModifiers, ChatMessage} -> send_0304(ChatTypeID, ChatGID, ChatName, ChatModifiers, ChatMessage), ?MODULE:loop(SoFar); {psu_keepalive} -> psu_proto:send_keepalive(get(socket)), ?MODULE:loop(SoFar); {psu_player_spawn, _Spawn} -> % Should be something along the lines of 203 201 204 or something. {ok, User} = egs_user_model:read(get(gid)), {ok, SpawnList} = egs_user_model:select({neighbors, User}), send_0233(SpawnList), ?MODULE:loop(SoFar); {psu_player_unspawn, Spawn} -> {ok, User} = egs_user_model:read(get(gid)), send_0204(User, Spawn, 5), ?MODULE:loop(SoFar); {psu_warp, QuestID, ZoneID, MapID, EntryID} -> event({area_change, QuestID, ZoneID, MapID, EntryID}), ?MODULE:loop(SoFar); {ssl, _, Data} -> {Packets, Rest} = psu_proto:packet_split(<< SoFar/bits, Data/bits >>), [dispatch(Orig) || Orig <- Packets], ?MODULE:loop(Rest); {ssl_closed, _} -> ssl_closed; %% exit {ssl_error, _, _} -> ssl_error; %% exit _ -> ?MODULE:loop(SoFar) after 1000 -> reload, ?MODULE:loop(SoFar) end. %% @doc Dispatch the command to the right handler. dispatch(Orig) -> case psu_proto:parse(Orig) of {command, Command, Channel, Data} -> case Channel of 1 -> broadcast(Command, Orig); _ -> handle(Command, Data) end; ignore -> ignore; Event -> event(Event) end. %% @doc Position change broadcast handler. Save the position and then dispatch it. broadcast(16#0503, Orig) -> << _: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, FloatDir = Dir / 46603.375, {ok, User} = egs_user_model:read(get(gid)), NewUser = User#egs_user_model{pos=#pos{x=X, y=Y, z=Z, dir=FloatDir}, area=#psu_area{questid=QuestID, zoneid=ZoneID, mapid=MapID}, entryid=EntryID}, egs_user_model:write(NewUser), broadcast(default, Orig); %% @doc Stand still broadcast handler. Save the position and then dispatch it. broadcast(16#0514, Orig) -> << _: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, MapID:32/little-unsigned-integer, EntryID:32/little-unsigned-integer, _/bits >> = Orig, FloatDir = Dir / 46603.375, {ok, User} = egs_user_model:read(get(gid)), NewUser = User#egs_user_model{pos=#pos{x=X, y=Y, z=Z, dir=FloatDir}, area=#psu_area{questid=QuestID, zoneid=ZoneID, mapid=MapID}, entryid=EntryID}, egs_user_model:write(NewUser), broadcast(default, Orig); %% @doc Default broadcast handler. Dispatch the command to everyone. %% 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. %% @todo Don't query the user data everytime! Keep an User instead of a GID probably. broadcast(Command, Orig) when Command =:= 16#0101; Command =:= 16#0102; Command =:= 16#0104; Command =:= 16#0107; Command =:= 16#010f; Command =:= 16#050f; Command =:= default -> << _:32, A:64/bits, _:64, B:192/bits, _:64, C/bits >> = Orig, GID = get(gid), case egs_user_model:read(GID) of {error, _Reason} -> ignore; {ok, Self} -> LID = Self#egs_user_model.lid, Packet = << A/binary, 16#00011300:32, GID:32/little-unsigned-integer, B/binary, GID:32/little-unsigned-integer, LID:32/little-unsigned-integer, C/binary >>, {ok, SpawnList} = egs_user_model:select({neighbors, Self}), lists:foreach(fun(User) -> User#egs_user_model.pid ! {psu_broadcast, Packet} end, SpawnList) end. %% @doc Trigger many events. events(Events) -> [event(Event) || Event <- Events], ok. %% @todo When changing lobby to the room, 0230 must also be sent. Same when going from room to lobby. %% @todo Probably move area_load inside the event and make other events call this one when needed. event({area_change, QuestID, ZoneID, MapID, EntryID}) -> log("area change (~b,~b,~b,~b)", [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 -> %% This probably shouldn't happen. Just make it crash on purpose. log("chat FromGID=0"), ignore; 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 [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}])]]), %% 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 There's at least 9 different sets of locations. Handle all of them correctly. event(counter_background_locations_request) -> send_170c(); %% @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. event({counter_enter, CounterID, FromZoneID, FromMapID, FromEntryID}) -> log("counter load ~b", [CounterID]), {ok, OldUser} = egs_user_model:read(get(gid)), OldArea = OldUser#egs_user_model.area, FromArea = {psu_area, OldArea#psu_area.questid, FromZoneID, FromMapID}, User = OldUser#egs_user_model{areatype=counter, area={psu_area, 16#7fffffff, 0, 0}, entryid=0, prev_area=FromArea, prev_entryid=FromEntryID}, egs_user_model:write(User), AreaName = "Counter", QuestFile = "data/lobby/counter.quest.nbl", ZoneFile = "data/lobby/counter.zone.nbl", %% broadcast unspawn to other people {ok, UnspawnList} = egs_user_model:select({neighbors, OldUser}), lists:foreach(fun(Other) -> Other#egs_user_model.pid ! {psu_player_unspawn, User} end, UnspawnList), %% load counter psu_proto:send_0c00(User), psu_proto:send_020e(User, QuestFile), psu_proto:send_0a05(User), psu_proto:send_010d(User, User#egs_user_model{lid=0}), send_0200(mission), send_020f(ZoneFile, 0, 16#ff), psu_proto:send_0205(User, 0), send_100e(16#7fffffff, 0, 0, AreaName, CounterID), psu_proto:send_0215(User, 0), psu_proto:send_0215(User, 0), psu_proto:send_020c(User), send_1202(), send_1204(), send_1206(), send_1207(), send_1212(), psu_proto:send_0201(User, User#egs_user_model{lid=0}), send_0a06(), case User#egs_user_model.partypid of undefined -> ignore; _ -> send_022c(0, 16#12) end, send_0208(), send_0236(); %% @doc Leave mission counter handler. event(counter_leave) -> {ok, User} = egs_user_model:read(get(gid)), PrevArea = User#egs_user_model.prev_area, event({area_change, PrevArea#psu_area.questid, PrevArea#psu_area.zoneid, PrevArea#psu_area.mapid, User#egs_user_model.prev_entryid}); %% @doc Send the code for the background image to use. But there's more that should be sent though. %% @todo Apparently background values 1 2 3 are never used on official servers. Find out why. event({counter_options_request, CounterID}) -> log("counter options request ~p", [CounterID]), [{quests, _}, {bg, Background}|_Tail] = proplists:get_value(CounterID, ?COUNTERS), send_1711(Background); %% @todo Handle when the party already exists! And stop doing it wrong. event(counter_party_info_request) -> {ok, User} = egs_user_model:read(get(gid)), send_1706((User#egs_user_model.character)#characters.name); %% @todo Item distribution is always set to random for now. event(counter_party_options_request) -> send_170a(); %% @doc Request the counter's quest files. event({counter_quest_files_request, CounterID}) -> log("counter quest files request ~p", [CounterID]), [{quests, Filename}|_Tail] = proplists:get_value(CounterID, ?COUNTERS), send_0c06(Filename); %% @doc Counter available mission list request handler. event({counter_quest_options_request, CounterID}) -> log("counter quest options request ~p", [CounterID]), [{quests, _}, {bg, _}, {options, Options}] = proplists:get_value(CounterID, ?COUNTERS), send_0c10(Options); %% @todo A and B are mostly unknown. Like most of everything else from the command 0e00... event({hit, FromTargetID, ToTargetID, A, B}) -> GID = get(gid), {ok, User} = egs_user_model:read(GID), %% hit! #hit_response{type=Type, user=NewUser, exp=HasEXP, damage=Damage, targethp=TargetHP, targetse=TargetSE, events=Events} = psu_instance:hit(User, FromTargetID, ToTargetID), case Type of box -> %% @todo also has a hit sent, we should send it too events(Events); _ -> PlayerHP = (NewUser#egs_user_model.character)#characters.currenthp, case lists:member(death, TargetSE) of true -> SE = 16#01000200; false -> SE = 16#01000000 end, send(<< 16#0e070300:32, 0:160, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, 1:32/little-unsigned-integer, 16#01050000:32, Damage:32/little-unsigned-integer, A/binary, 0:64, PlayerHP:32/little-unsigned-integer, 0:32, SE:32, 0:32, TargetHP:32/little-unsigned-integer, 0:32, B/binary, 16#04320000:32, 16#80000000:32, 16#26030000:32, 16#89068d00:32, 16#0c1c0105:32, 0:64 >>) % after TargetHP is SE-related too? end, %% exp if HasEXP =:= true -> Character = NewUser#egs_user_model.character, Level = Character#characters.mainlevel, send_0115(GID, ToTargetID, Level#level.number, Level#level.exp, Character#characters.money); true -> ignore end, %% save egs_user_model:write(NewUser); event({hits, Hits}) -> [event(Hit) || Hit <- Hits], ok; %% @todo Send something other than just "dammy". event({item_description_request, ItemID}) -> case proplists:get_value(ItemID, ?ITEMS) of undefined -> send_0a11(ItemID, "Always bet on Dammy."); #psu_item{description=Desc} -> send_0a11(ItemID, Desc) end; %% @todo A and B are unknown. %% Melee uses a format similar to: AAAA--BBCCCC----DDDDDDDDEE----FF with %% AAAA the attack sound effect, BB the range, CCCC and DDDDDDDD unknown but related to angular range or similar, EE number of targets and FF the model. %% Bullets and tech weapons formats are unknown but likely use a slightly different format. %% @todo Others probably want to see that you changed your weapon. %% @todo Apparently B is always ItemID+1. Not sure why. %% @todo Currently use a separate file for the data sent for the weapons. %% @todo TargetGID and TargetLID must be validated, they're either the player's or his NPC characters. event({item_equip, ItemIndex, TargetGID, TargetLID, A, B}) -> GID = get(gid), Category = case ItemIndex of % units would be 8, traps would be 12 19 -> 2; % armor Y when Y =:= 5; Y =:= 6; Y =:= 7 -> 0; % clothes _ -> 1 % weapons end, Filename = case ItemIndex of % weapons 16 -> "p/packet0105_sword.bin"; 13 -> "p/packet0105_twindaggers.bin"; 15 -> "p/packet0105_dagger.bin"; 9 -> "p/packet0105_rcsm.bin"; 14 -> "p/packet0105_saber.bin"; 8 -> "p/packet0105_mgun.bin"; X when X =:= 17; X =:= 18 -> "p/packet0105_twinguns.bin"; % armor 19 -> "p/packet0105_armor.bin"; % clothes X when X =:= 5; X =:= 6; X =:= 7 -> none; _ -> % default, for lou "p/packet0105_twindaggers.bin" end, case Filename of none -> File = << >>; _ -> {ok, File} = file:read_file(Filename) end, send(<< 16#01050300:32, 0:64, TargetGID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, TargetGID:32/little-unsigned-integer, TargetLID:32/little-unsigned-integer, ItemIndex, 1, Category, A, B:32/little-unsigned-integer, File/binary >>); %% @todo A and B are unknown. %% @see item_equip event({item_unequip, ItemIndex, TargetGID, TargetLID, A, B}) -> GID = get(gid), Category = case ItemIndex of % units would be 8, traps would be 12 19 -> 2; % armor Y when Y =:= 5; Y =:= 6; Y =:= 7 -> 0; % clothes _ -> 1 % weapons end, send(<< 16#01050300:32, 0:64, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, TargetGID:32/little-unsigned-integer, TargetLID:32/little-unsigned-integer, ItemIndex, 2, Category, A, B:32/little-unsigned-integer >>); %% @todo Just ignore the meseta price for now and send the player where he wanna be! event(lobby_transport_request) -> send_0c08(true); %% @todo Handle all different Lumilass. event(lumilass_options_request) -> send_1a03(); %% @todo Probably replenish the player HP when entering a non-mission area rather than when aborting the mission? event(mission_abort) -> send_1006(11, 0), {ok, User} = egs_user_model:read(get(gid)), %% delete the mission if User#egs_user_model.instancepid =:= undefined -> ignore; true -> psu_instance:stop(User#egs_user_model.instancepid) end, %% full hp Character = User#egs_user_model.character, MaxHP = Character#characters.maxhp, NewCharacter = Character#characters{currenthp=MaxHP}, NewUser = User#egs_user_model{character=NewCharacter, setid=0}, egs_user_model:write(NewUser), %% map change if User#egs_user_model.areatype =:= mission -> PrevArea = User#egs_user_model.prev_area, event({area_change, PrevArea#psu_area.questid, PrevArea#psu_area.zoneid, PrevArea#psu_area.mapid, User#egs_user_model.prev_entryid}); true -> ignore end; %% @todo Forward the mission start to other players of the same party, whatever their location is. event({mission_start, QuestID}) -> log("mission start ~b", [QuestID]), send_1020(), send_1015(QuestID), send_0c02(); %% @doc Force the invite of an NPC character while inside a mission. Mostly used by story missions. %% Note that the NPC is often removed and reinvited between block/cutscenes. event({npc_force_invite, NPCid}) -> GID = get(gid), {ok, User} = egs_user_model:read(GID), %% Create NPC. log("npc force invite ~p", [NPCid]), TmpNPCUser = psu_npc:user_init(NPCid, ((User#egs_user_model.character)#characters.mainlevel)#level.number), %% Create and join party. case User#egs_user_model.partypid of undefined -> {ok, PartyPid} = psu_party:start_link(GID); PartyPid -> ignore end, {ok, PartyPos} = psu_party:join(PartyPid, npc, TmpNPCUser#egs_user_model.id), #egs_user_model{instancepid=InstancePid, area=Area, entryid=EntryID, pos=Pos} = User, NPCUser = TmpNPCUser#egs_user_model{lid=PartyPos, partypid=PartyPid, instancepid=InstancePid, areatype=mission, area=Area, entryid=EntryID, pos=Pos}, egs_user_model:write(NPCUser), egs_user_model:write(User#egs_user_model{partypid=PartyPid}), %% Send stuff. Character = NPCUser#egs_user_model.character, SentNPCCharacter = Character#characters{gid=NPCid, npcid=NPCid}, SentNPCUser = NPCUser#egs_user_model{character=SentNPCCharacter}, psu_proto:send_010d(User, SentNPCUser), psu_proto:send_0201(User, SentNPCUser), psu_proto:send_0215(User, 0), send_0a04(SentNPCUser#egs_user_model.id), send_022c(0, 16#12), send_1004(npc_mission, SentNPCUser, PartyPos), send_100f((SentNPCUser#egs_user_model.character)#characters.npcid, PartyPos), send_1601(PartyPos); %% @todo Also at the end send a 101a (NPC:16, PartyPos:16, ffffffff). Not sure about PartyPos. event({npc_invite, NPCid}) -> GID = get(gid), {ok, User} = egs_user_model:read(GID), %% Create NPC. log("invited npcid ~b", [NPCid]), TmpNPCUser = psu_npc:user_init(NPCid, ((User#egs_user_model.character)#characters.mainlevel)#level.number), %% Create and join party. case User#egs_user_model.partypid of undefined -> {ok, PartyPid} = psu_party:start_link(GID), send_022c(0, 16#12); PartyPid -> ignore end, {ok, PartyPos} = psu_party:join(PartyPid, npc, TmpNPCUser#egs_user_model.id), NPCUser = TmpNPCUser#egs_user_model{lid=PartyPos, partypid=PartyPid}, egs_user_model:write(NPCUser), egs_user_model:write(User#egs_user_model{partypid=PartyPid}), %% Send stuff. Character = NPCUser#egs_user_model.character, SentNPCCharacter = Character#characters{gid=NPCid, npcid=NPCid}, SentNPCUser = NPCUser#egs_user_model{character=SentNPCCharacter}, send_1004(npc_invite, SentNPCUser, PartyPos), send_101a(NPCid, PartyPos); %% @todo Should be 0115(money) 010a03(confirm sale). %% @todo We probably need to save the ShopID somewhere since it isn't given back here. event({npc_shop_buy, ShopItemIndex, Quantity}) -> log("npc shop buy itemindex ~p quantity ~p", [ShopItemIndex, Quantity]); %% @todo Currently send the normal items shop for all shops, differentiate. event({npc_shop_enter, ShopID}) -> log("npc shop enter ~p", [ShopID]), case proplists:get_value(ShopID, ?SHOPS) of undefined -> %% @todo Temporary; prevent players from getting stuck. GID = get(gid), {ok, File} = file:read_file("p/itemshop.bin"), send(<< 16#010a0300: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, File/binary >>); ItemsList -> send_010a(ItemsList) end; event({npc_shop_leave, ShopID}) -> log("npc shop leave ~p", [ShopID]), GID = get(gid), send(<< 16#010a0300: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 >>); %% @todo Should be 0115(money) 010a03(confirm sale). event({npc_shop_sell, InventoryItemIndex, Quantity}) -> log("npc shop sell itemindex ~p quantity ~p", [InventoryItemIndex, Quantity]); %% @todo First 1a02 value should be non-0. %% @todo Could the 2nd 1a02 parameter simply be the shop type or something? %% @todo Although the values replied should be right, they seem mostly ignored by the client. event({npc_shop_request, ShopID}) -> log("npc shop request ~p", [ShopID]), case ShopID of 80 -> send_1a02(0, 17, 17, 3, 9); %% lumilass 90 -> send_1a02(0, 5, 1, 4, 5); %% parum weapon grinding 91 -> send_1a02(0, 5, 5, 4, 7); %% tenora weapon grinding 92 -> send_1a02(0, 5, 0, 4, 0); %% yohmei weapon grinding 93 -> send_1a02(0, 5, 18, 4, 0); %% kubara weapon grinding _ -> send_1a02(0, 0, 1, 0, 0) end; %% @todo Not sure what are those hardcoded values. event({object_boss_gate_activate, ObjectID}) -> send_1213(ObjectID, 0), send_1215(2, 16#7008), %% @todo Following sent after the warp? send_1213(37, 0), %% @todo Why resend this? send_1213(ObjectID, 0); event({object_boss_gate_enter, ObjectID}) -> send_1213(ObjectID, 1); %% @todo Do we need to send something back here? event({object_boss_gate_leave, _ObjectID}) -> ignore; %% @doc Server-side event. event({object_box_destroy, ObjectID}) -> send_1213(ObjectID, 3); %% @todo Second send_1211 argument should be User#egs_user_model.lid. Fix when it's correctly handled. event({object_chair_sit, ObjectTargetID}) -> %~ {ok, User} = egs_user_model:read(get(gid)), send_1211(ObjectTargetID, 0, 8, 0); %% @todo Second send_1211 argument should be User#egs_user_model.lid. Fix when it's correctly handled. event({object_chair_stand, ObjectTargetID}) -> %~ {ok, User} = egs_user_model:read(get(gid)), send_1211(ObjectTargetID, 0, 8, 2); event({object_crystal_activate, ObjectID}) -> send_1213(ObjectID, 1); %% @doc Server-side event. event({object_event_trigger, BlockID, EventID}) -> send_1205(EventID, BlockID, 0); event({object_key_console_enable, ObjectID}) -> {ok, User} = egs_user_model:read(get(gid)), {BlockID, [EventID|_]} = psu_instance:std_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, ObjectID), send_1205(EventID, BlockID, 0), send_1213(ObjectID, 1); event({object_key_console_init, ObjectID}) -> {ok, User} = egs_user_model:read(get(gid)), {BlockID, [_, EventID, _]} = psu_instance:std_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, ObjectID), send_1205(EventID, BlockID, 0); event({object_key_console_open_gate, ObjectID}) -> {ok, User} = egs_user_model:read(get(gid)), {BlockID, [_, _, EventID]} = psu_instance:std_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, ObjectID), send_1205(EventID, BlockID, 0), send_1213(ObjectID, 1); %% @todo Now that it's separate from object_key_console_enable, handle it better than that, don't need a list of events. event({object_key_enable, ObjectID}) -> {ok, User} = egs_user_model:read(get(gid)), {BlockID, [EventID|_]} = psu_instance:std_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, ObjectID), send_1205(EventID, BlockID, 0), send_1213(ObjectID, 1); %% @todo Some switch objects apparently work differently, like the light switch in Mines in MAG'. event({object_switch_off, ObjectID}) -> {ok, User} = egs_user_model:read(get(gid)), {BlockID, EventID} = psu_instance:std_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, ObjectID), send_1205(EventID, BlockID, 1), send_1213(ObjectID, 0); event({object_switch_on, ObjectID}) -> {ok, User} = egs_user_model:read(get(gid)), {BlockID, EventID} = psu_instance:std_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, ObjectID), send_1205(EventID, BlockID, 0), send_1213(ObjectID, 1); event({object_vehicle_boost_enable, ObjectID}) -> send_1213(ObjectID, 1); event({object_vehicle_boost_respawn, ObjectID}) -> send_1213(ObjectID, 0); %% @todo Second send_1211 argument should be User#egs_user_model.lid. Fix when it's correctly handled. event({object_warp_take, BlockID, ListNb, ObjectNb}) -> {ok, User} = egs_user_model:read(get(gid)), Pos = psu_instance:warp_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, BlockID, ListNb, ObjectNb), NewUser = User#egs_user_model{pos=Pos}, egs_user_model:write(NewUser), send_0503(User#egs_user_model.pos), send_1211(16#ffffffff, 0, 14, 0); event({party_remove_member, PartyPos}) -> log("party remove member ~b", [PartyPos]), {ok, DestUser} = egs_user_model:read(get(gid)), {ok, RemovedGID} = psu_party:get_member(DestUser#egs_user_model.partypid, PartyPos), psu_party:remove_member(DestUser#egs_user_model.partypid, PartyPos), {ok, RemovedUser} = egs_user_model:read(RemovedGID), case (RemovedUser#egs_user_model.character)#characters.type of npc -> egs_user_model:delete(RemovedGID); _ -> ignore end, send_1006(8, PartyPos), send_0204(DestUser, RemovedUser, 1), psu_proto:send_0215(DestUser, 0); event({player_options_change, Options}) -> {ok, User} = egs_user_model:read(get(gid)), file:write_file(io_lib:format("save/~s/~b-character.options", [User#egs_user_model.folder, (User#egs_user_model.character)#characters.slot]), Options); %% @todo If the player has a scape, use it! Otherwise red screen. %% @todo Right now we force revive and don't update the player's HP. event(player_death) -> % @todo send_0115(get(gid), 16#ffffffff, LV=1, EXP=idk, Money=1000), % apparently sent everytime you die... %% use scape: NewHP = 10, send_0117(NewHP), send_1022(NewHP); %% red screen with return to lobby choice: %~ send_0111(3, 1); %% @todo Refill the player's HP to maximum, remove SEs etc. event(player_death_return_to_lobby) -> {ok, User} = egs_user_model:read(get(gid)), PrevArea = User#egs_user_model.prev_area, event({area_change, PrevArea#psu_area.questid, PrevArea#psu_area.zoneid, PrevArea#psu_area.mapid, User#egs_user_model.prev_entryid}); event(player_type_availability_request) -> send_1a07(); event(player_type_capabilities_request) -> send_0113(); event(ppcube_request) -> send_1a04(); %% @doc Uni cube handler. event(unicube_request) -> send_021e(); %% @doc Uni selection handler. %% @todo When selecting 'Your room', load a default room. %% @todo When selecting 'Reload', reload the character in the current lobby. %% @todo Delete NPC characters and stop the party on entering myroom too. event({unicube_select, Selection, EntryID}) -> case Selection of cancel -> ignore; 16#ffffffff -> log("uni selection (my room)"), send_0230(), % 0220 event({area_change, 1120000, 0, 100, 0}); _UniID -> log("uni selection (reload)"), send_0230(), % 0220 %% force reloading the character and data files (@todo hack, uses myroom questid to do it) {ok, User} = egs_user_model:read(get(gid)), case User#egs_user_model.partypid of undefined -> ignore; PartyPid -> %% @todo Replace stop by leave when leaving stops the party correctly when nobody's there anymore. %~ psu_party:leave(User#egs_user_model.partypid, User#egs_user_model.id) {ok, NPCList} = psu_party:get_npc(PartyPid), [egs_user_model:delete(NPCGID) || {_Spot, NPCGID} <- NPCList], psu_party:stop(PartyPid) end, Area = User#egs_user_model.area, NewRow = User#egs_user_model{partypid=undefined, area=Area#psu_area{questid=1120000, zoneid=undefined}, entryid=EntryID}, egs_user_model:write(NewRow), event({area_change, Area#psu_area.questid, Area#psu_area.zoneid, Area#psu_area.mapid, EntryID}) end. %% @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 Type shouldn't be :32 but it seems when the later 16 have something it's not a spawn event. handle(16#0402, Data) -> << SpawnID:32/little-unsigned-integer, _:64, Type:32/little-unsigned-integer, _:64 >> = Data, case Type of 7 -> % spawn cleared @todo 1201 sent back with same values apparently, but not always log("cleared spawn ~b", [SpawnID]), {ok, User} = egs_user_model:read(get(gid)), {BlockID, EventID} = psu_instance:spawn_cleared_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, SpawnID), if EventID =:= false -> ignore; true -> send_1205(EventID, BlockID, 0) end; _ -> ignore end; %% @todo Handle this packet. %% @todo 3rd Unsafe Passage C, EventID 10 BlockID 2 = mission cleared? handle(16#0404, Data) -> << EventID:8, BlockID:8, _:16, Value:8, _/bits >> = Data, log("unknown command 0404: eventid ~b blockid ~b value ~b", [EventID, BlockID, Value]), send_1205(EventID, BlockID, Value); %% @todo Used in the tutorial. Not sure what it does. Give an item (the PA) maybe? handle(16#0a09, Data) -> log("0a09 ~p", [Data]), GID = get(gid), send(<< 16#0a090300:32, 0:32, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, 16#00003300:32, 0:32 >>); %% @todo Figure out this command. handle(16#0c11, << A:32/little, B:32/little >>) -> log("0c11 ~p ~p", [A, B]), GID = get(gid), send(<< 16#0c120300:32, 0:160, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, A:32/little, 1:32/little >>); %% @doc Set flag handler. Associate a new flag with the character. %% Just reply with a success value for now. %% @todo God save the flags. handle(16#0d04, Data) -> << Flag:128/bits, A:16/bits, _:8, B/bits >> = Data, log("flag handler for ~s", [re:replace(Flag, "\\0+", "", [global, {return, binary}])]), GID = get(gid), send(<< 16#0d040300:32, 0:160, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, Flag/binary, A/binary, 1, B/binary >>); %% @doc Initialize a vehicle object. %% @todo Find what are the many values, including the odd Whut value (and whether it's used in the reply). %% @todo Separate the reply. handle(16#0f00, Data) -> << A:32/little-unsigned-integer, 0:16, B:16/little-unsigned-integer, 0:16, C:16/little-unsigned-integer, 0, Whut:8, D:16/little-unsigned-integer, 0:16, E:16/little-unsigned-integer, 0:16, F:16/little-unsigned-integer, G:16/little-unsigned-integer, H:16/little-unsigned-integer, I:32/little-unsigned-integer >> = Data, log("init vehicle: ~b ~b ~b ~b ~b ~b ~b ~b ~b ~b", [A, B, C, Whut, D, E, F, G, H, I]), send(<< (header(16#1208))/binary, A:32/little-unsigned-integer, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 16#ffffffff:32, 0:16, B:16/little-unsigned-integer, 0:16, C:16/little-unsigned-integer, 0:16, D:16/little-unsigned-integer, 0:112, E:16/little-unsigned-integer, 0:16, F:16/little-unsigned-integer, H:16/little-unsigned-integer, 1, 0, 100, 0, 10, 0, G:16/little-unsigned-integer, 0:16 >>); %% @doc Enter vehicle. %% @todo Separate the reply. handle(16#0f02, Data) -> << A:32/little-unsigned-integer, B:32/little-unsigned-integer, C:32/little-unsigned-integer >> = Data, log("enter vehicle: ~b ~b ~b", [A, B, C]), HP = 100, send(<< (header(16#120a))/binary, A:32/little-unsigned-integer, B:32/little-unsigned-integer, C:32/little-unsigned-integer, HP:32/little-unsigned-integer >>); %% @doc Sent right after entering the vehicle. Can't move without it. %% @todo Separate the reply. handle(16#0f07, Data) -> << A:32/little-unsigned-integer, B:32/little-unsigned-integer >> = Data, log("after enter vehicle: ~b ~b", [A, B]), send(<< (header(16#120f))/binary, A:32/little-unsigned-integer, B:32/little-unsigned-integer >>); %% @todo Not sure yet. handle(16#1019, _) -> ignore; %~ send(<< (header(16#1019))/binary, 0:192, 16#00200000:32, 0:32 >>); %% @todo Not sure about that one though. Probably related to 1112 still. handle(16#1106, Data) -> send_110e(Data); %% @doc Probably asking permission to start the video (used for syncing?). handle(16#1112, Data) -> send_1113(Data); %% @todo Not sure yet. Value is probably a TargetID. Used in Airboard Rally. Replying with the same value starts the race. handle(16#1216, Data) -> << Value:32/little-unsigned-integer >> = Data, log("command 1216 with value ~b", [Value]), send_1216(Value); %% @doc Unknown command handler. Do nothing. handle(Command, _) -> log("dismissed packet ~4.16.0b", [Command]). %% @doc Build the packet header. header(Command) -> GID = get(gid), << 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. %% @todo Consolidate the receive and send functions better. send(Packet) -> psu_proto:packet_send(get(socket), Packet). %% @doc Send a shop listing. send_010a(ItemsList) -> GID = get(gid), NbItems = length(ItemsList), ItemsBin = build_010a_list(ItemsList, []), send(<< 16#010a0300:32, 0:64, GID:32/little, 0:64, 16#00011300:32, GID:32/little, 0:64, GID:32/little, 0:32, 1:16/little, NbItems:8, 2:8, 0:32, ItemsBin/binary >>). %% @todo The values set to 0 are unknown. build_010a_list([], Acc) -> iolist_to_binary(lists:reverse(Acc)); build_010a_list([ItemID|Tail], Acc) -> #psu_item{name=Name, rarity=Rarity, buy_price=SellPrice, data=Data} = proplists:get_value(ItemID, ?ITEMS), UCS2Name = << << X:8, 0:8 >> || X <- Name >>, NamePadding = 8 * (46 - byte_size(UCS2Name)), RarityBin = Rarity - 1, DataBin = build_010a_item(Data), BinItemID = case element(1, Data) of psu_clothing_item -> %% Change the ItemID to enable all colors. << A:8, _:4, B:12, _:8 >> = << ItemID:32 >>, << A:8, 3:4, B:12, 16#ff:8 >>; _Any -> << ItemID:32 >> end, Bin = << UCS2Name/binary, 0:NamePadding, RarityBin:8, 0:8, BinItemID/binary, SellPrice:32/little, DataBin/binary >>, build_010a_list(Tail, [Bin|Acc]). build_010a_item(#psu_clothing_item{appearance=Appearance, manufacturer=Manufacturer, type=Type, overlap=Overlap, gender=Gender, colors=Colors}) -> GenderInt = case Gender of male -> 16#1b; female -> 16#2b end, << Appearance:16, Type:4, Manufacturer:4, Overlap:8, GenderInt:8, Colors/binary, 0:40 >>; build_010a_item(#psu_consumable_item{max_quantity=MaxQuantity, pt_diff=PointsDiff, status_effect=StatusEffect, target=Target, use_condition=UseCondition, item_effect=ItemEffect}) -> << 0:8, MaxQuantity:8, Target:8, UseCondition:8, PointsDiff:16/little, StatusEffect:8, ItemEffect:8, 0:96 >>; build_010a_item(#psu_parts_item{appearance=Appearance, manufacturer=Manufacturer, type=Type, overlap=Overlap, gender=Gender}) -> GenderInt = case Gender of male -> 16#14; female -> 16#24 end, << Appearance:16, Type:4, Manufacturer:4, Overlap:8, GenderInt:8, 0:120 >>; build_010a_item(#psu_trap_item{max_quantity=MaxQuantity}) -> << 2:32/little, 16#ffffff:24, MaxQuantity:8, 0:96 >>. %% @doc Trigger a character-related event. send_0111(A, B) -> 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, GID:32/little-unsigned-integer, 0:32, A:32/little-unsigned-integer, B:32/little-unsigned-integer >>). %% @todo Types capability list. send_0113() -> {ok, File} = file:read_file("p/typesinfo.bin"), 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 >>). %% @doc Update the character's EXP, level or 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, 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, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32 >>). %% @doc Revive player? %% @todo Figure out more of it. send_0117(HP) -> 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, GID:32/little-unsigned-integer, 0:96, HP:32/little-unsigned-integer, 0:32 >>). %% @doc Send the zone initialization notification. send_0200(ZoneType) -> case ZoneType of mission -> Var = << 16#06000500:32, 16#01000000:32, 0:64, 16#00040000:32, 16#00010000:32, 16#00140000:32 >>; myroom -> Var = << 16#06000000:32, 16#02000000:32, 0:64, 16#40000000:32, 16#00010000:32, 16#00010000:32 >>; _ -> Var = << 16#00040000:32, 0:160, 16#00140000:32 >> end, send(<< (header(16#0200))/binary, 0:32, 16#01000000:32, 16#ffffffff:32, Var/binary, 16#ffffffff:32, 16#ffffffff:32 >>). %% @todo Not sure. Used for unspawning, and more. send_0204(DestUser, TargetUser, Action) -> DestGID = DestUser#egs_user_model.id, TargetTypeID = case (TargetUser#egs_user_model.character)#characters.type of npc -> 16#00001d00; _ -> 16#00001200 end, #egs_user_model{id=TargetGID, lid=TargetLID} = TargetUser, send(<< 16#02040300:32, 0:32, TargetTypeID:32, TargetGID:32/little-unsigned-integer, 0:64, 16#00011300:32, DestGID:32/little-unsigned-integer, 0:64, TargetGID:32/little-unsigned-integer, TargetLID:32/little-unsigned-integer, Action:32/little-unsigned-integer >>). %% @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. send_0208() -> send(<< (header(16#0208))/binary, 2:32/little-unsigned-integer >>). %% @doc Send the zone file to be loaded. send_020f(Filename, SetID, SeasonID) -> {ok, File} = file:read_file(Filename), Size = byte_size(File), send(<< 16#020f0300:32, 16#ffff:16, 0:272, SetID, SeasonID, 0:16, Size:32/little-unsigned-integer, File/binary >>). %% @doc Send the current UNIX time. send_0210() -> GID = get(gid), CurrentTime = calendar:datetime_to_gregorian_seconds(calendar:now_to_universal_time(now())) - calendar:datetime_to_gregorian_seconds({{1970, 1, 1}, {0, 0, 0}}), send(<< 16#02100300:32, 16#ffff:16, 0:144, 16#00011300:32, GID:32/little, 0:96, CurrentTime:32/little-unsigned-integer >>). %% @todo End of character loading. Just send it. send_021b() -> GID = get(gid), send(<< 16#021b0300:32, 16#ffff:16, 0:144, 16#00011300:32, GID:32/little, 0:64 >>). %% @doc Send the list of available universes. send_021e() -> {ok, Count} = egs_user_model:count(), [StrCount] = io_lib:format("~b", [Count]), Unis = [{16#ffffffff, center, "Your Room", ""}, {1, justify, "Reload", " "}, {2, justify, "EGS Test", StrCount}], NbUnis = length(Unis), Bin = send_021e_build(Unis, []), send(<< 16#021e0300:32, 0:288, NbUnis:32/little-unsigned-integer, Bin/binary >>). send_021e_build([], Acc) -> iolist_to_binary(lists:reverse(Acc)); send_021e_build([{ID, Align, Name, Pop}|Tail], Acc) -> UCS2Name = << << X:8, 0:8 >> || X <- Name >>, UCS2Pop = << << X:8, 0:8 >> || X <- Pop >>, NamePadding = 8 * (32 - byte_size(UCS2Name)), PopPadding = 8 * (12 - byte_size(UCS2Pop)), IntAlign = case Align of justify -> 643; center -> 0 end, send_021e_build(Tail, [<< ID:32/little-unsigned-integer, 0:16, IntAlign:16/little-unsigned-integer, UCS2Name/binary, 0:NamePadding, UCS2Pop/binary, 0:PopPadding >>|Acc]). %% @doc Send the current universe name and number. %% @todo Currently only have universe number 2, named EGS Test. %% @todo We must have a parameter indicating whether this is a room or a normal universe. send_0222() -> UCS2Name = << << X:8, 0:8 >> || X <- "EGS Test" >>, Padding = 8 * (44 - byte_size(UCS2Name)), UniID = 2, GID = get(gid), send(<< 16#02220300:32, 16#ffff:16, 0:16, 16#00001200:32, GID:32/little, 0:64, 16#00011300:32, GID:32/little, 0:64, UniID:32/little, 16#01009e02:32, UCS2Name/binary, 0:Padding, 16#aa000000:32 >>). %% @todo No idea! send_022c(A, B) -> 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. send_0230() -> 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 >>). %% @todo Figure out what the other things are. send_0233(Users) -> NbUsers = length(Users), case NbUsers of 0 -> ignore; _ -> GID = get(gid), Header = << 16#02330300:32, 0:32, 16#00001200:32, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, NbUsers:32/little-unsigned-integer >>, Contents = build_0233_contents(Users), send(<< Header/binary, Contents/binary >>) end. %% @todo God this function is ugly. Use tail recursion! build_0233_contents([]) -> << >>; build_0233_contents(Users) -> [User|Rest] = Users, LID = 16#010000 + User#egs_user_model.lid, %% @todo The LID must be 16 bits and 0233 seems to (almost always) require that 01 right there... CharBin = psu_characters:character_user_to_binary(User#egs_user_model{lid=LID}), IsGM = 0, GameVersion = 0, Chunk = << CharBin/binary, IsGM:8, 0:8, GameVersion:8, 0:8 >>, Next = build_0233_contents(Rest), << Chunk/binary, Next/binary >>. %% @doc Center the camera on the player, if possible. %% @todo Probably. send_0236() -> send(header(16#0236)). %% @doc Send a chat command. send_0304(FromTypeID, FromGID, FromName, Modifiers, Message) -> {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 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=_}) -> {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, IntDir = trunc(Dir * 182.0416), send(<< 16#05030300: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, 16#1000:16, IntDir:16/little-unsigned-integer, PrevX:32/little-float, PrevY:32/little-float, PrevZ:32/little-float, 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, 1:32/little-unsigned-integer >>). %% @todo NPC inventory. Guessing it's only for NPC characters... send_0a04(NPCGID) -> GID = get(gid), {ok, Bin} = file:read_file("p/packet0a04.bin"), send(<< 16#0a040300:32, 0:32, 16#00001d00:32, NPCGID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, Bin/binary >>). %% @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? send_0a06() -> {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), 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. send_0a0a() -> {ok, << _:32, A:224/bits, _:32, B/bits >>} = file:read_file("p/packet0a0a.bin"), GID = get(gid), send(<< A/binary, GID:32/little-unsigned-integer, B/binary >>). %% @doc Item description. send_0a11(ItemID, ItemDesc) -> Size = 1 + length(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 >>). %% @todo Figure out last 4 bytes! send_0c02() -> send(<< (header(16#0c02))/binary, 0:32 >>). %% @doc Send the huge pack of quest files available in the counter. send_0c06(Filename) -> {ok, << File/bits >>} = file:read_file(Filename), 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. %% Use true for allowing it, and false otherwise. send_0c08(Response) -> Value = if Response =:= true -> 0; true -> 1 end, send(<< (header(16#0c08))/binary, Value:32 >>). %% @doc Send the trial start notification. send_0c09() -> send(<< (header(16#0c09))/binary, 0:64 >>). %% @doc Send the counter's mission options (0 = invisible, 2 = disabled, 3 = available). send_0c10(Options) -> GID = get(gid), send(<< 16#0c100300:32, 0:32, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, Options/binary >>). %% @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 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) -> GID = User#egs_user_model.id, 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), send(<< 16#0d010300:32, 16#ffff:16, 0:144, 16#00011300:32, GID:32/little, 0:64, CharBin/binary, 16#ffbbef1c:32, 16#f8ff0700:32, 16#fc810916:32, 16#7802134c:32, 16#b0c0040f:32, 16#7cf0e583:32, 16#b7bce0c6:32, 16#7ff8f963:32, 16#3fd7ffff:32, 16#fff7ffff:32, 16#f3ff63e0:32, 16#1fe00000:32, 0:7744, OptionsBin/binary >>). %% @doc Send the character list for selection. %% @todo There's a few odd values blanked, also the last known location apparently. send_0d03(Data0, Data1, Data2, Data3) -> [{status, Status0}, {char, Char0}|_] = Data0, [{status, Status1}, {char, Char1}|_] = Data1, [{status, Status2}, {char, Char2}|_] = Data2, [{status, Status3}, {char, Char3}|_] = Data3, GID = get(gid), send(<< 16#0d030300:32/unsigned-integer, 0:32, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, 0:104, Status0:8/unsigned-integer, 0:48, Char0/binary, 0:520, Status1:8/unsigned-integer, 0:48, Char1/binary, 0:520, Status2:8/unsigned-integer, 0:48, Char2/binary, 0:520, Status3:8/unsigned-integer, 0:48, Char3/binary, 0:512 >>). %% @todo Add a character (NPC or real) to the party members on the right of the screen. %% @todo NPCid is 65535 for normal characters. %% @todo Apparently the 4 location ids are set to 0 when inviting an NPC in the lobby - NPCs have their location set to 0 when in lobby; also odd value before PartyPos related to missions %% @todo Not sure about LID. But seems like it. send_1004(Type, User, PartyPos) -> [TypeID, LID, SomeFlag] = case Type of npc_mission -> [16#00001d00, PartyPos, 2]; npc_invite -> [0, 16#ffffffff, 3]; _ -> 1 %% seems to be for players end, UserGID = get(gid), #egs_user_model{id=GID, character=Character, area={psu_area, QuestID, ZoneID, MapID}, entryid=EntryID} = User, #characters{npcid=NPCid, name=Name, mainlevel=MainLevel} = Character, Level = MainLevel#level.number, send(<< 16#10040300:32, 16#ffff0000:32, 0:128, 16#00011300:32, UserGID:32/little-unsigned-integer, 0:64, TypeID:32, GID:32/little-unsigned-integer, 0:64, Name/binary, Level:16/little-unsigned-integer, 16#ffff:16, SomeFlag, 1, PartyPos:8, 1, NPCid:16/little-unsigned-integer, 0:16, %% Odd unknown values. PA related? No idea. Values on invite, 0 in-mission. %~ 16#00001f08:32, 0:32, 16#07000000:32, %~ 16#04e41f08:32, 0:32, 16#01000000:32, %~ 16#64e41f08:32, 0:32, 16#02000000:32, %~ 16#64e41f08:32, 0:32, 16#03000000:32, %~ 16#64e41f08:32, 0:32, 16#12000000:32, %~ 16#24e41f08:32, 0:512, QuestID:32/little-unsigned-integer, ZoneID:32/little-unsigned-integer, MapID:32/little-unsigned-integer, EntryID:32/little-unsigned-integer, LID:32/little, 0:64, 16#01000000:32, 16#01000000:32, %% @todo first is current hp, second is max hp 0:608 >>). %% @todo Figure out what the packet is. send_1005(Name) -> {ok, File} = file:read_file("p/packet1005.bin"), << _:352, Before:160/bits, _:608, After/bits >> = File, GID = get(gid), send(<< 16#10050300:32, 16#ffff:16, 0:144, 16#00011300:32, GID:32/little, 0:64, Before/binary, GID:32/little, 0:64, Name/binary, After/binary >>). %% @doc Party-related command probably controlling the party state. %% EventID 11 aborts the mission. send_1006(EventID, PartyPos) -> GID = get(gid), send(<< 16#10060300:32, 16#ffff:16, 0:144, 16#00011300:32, GID:32/little, 0:64, EventID:8, PartyPos:8, 0:16 >>). %% @doc Send the player's current location. send_100e(QuestID, ZoneID, MapID, Location, CounterID) -> GID = get(gid), UCS2Location = << << X:8, 0:8 >> || X <- Location >>, Packet = << 16#100e0300:32, 16#ffffffbf:32, 0:128, 16#00011300:32, GID:32/little, 0:64, 1:32/little, MapID:16/little, ZoneID:16/little, QuestID:32/little, UCS2Location/binary >>, PaddingSize = (128 - byte_size(Packet) - 8) * 8, case CounterID of 16#ffffffff -> Footer = << CounterID:32/little-unsigned-integer, 0:32 >>; _ -> Footer = << CounterID:32/little-unsigned-integer, 1:32/little-unsigned-integer >> end, send(<< Packet/binary, 0:PaddingSize, Footer/binary >>). %% @todo No idea. Also the 2 PartyPos in the built packet more often than not match, but sometimes don't? That's probably because one is PartyPos and the other is LID or something. send_100f(NPCid, PartyPos) -> send(<< (header(16#100f))/binary, NPCid:16/little-unsigned-integer, 1, PartyPos:8, PartyPos:32/little-unsigned-integer >>). %% @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. send_1015(QuestID) -> [{type, _}, {file, QuestFile}|_] = proplists:get_value(QuestID, ?QUESTS), {ok, File} = file:read_file(QuestFile), Size = byte_size(File), send(<< (header(16#1015))/binary, QuestID:32/little-unsigned-integer, 16#01010000:32, 0:32, Size:32/little-unsigned-integer, File/binary >>). %% @todo No idea. send_1016(PartyPos) -> GID = get(gid), send(<< 16#10160300:32, 16#ffff0000:32, 0:128, 16#00011300:32, GID:32/little, 0:64, PartyPos:32/little >>). %% @todo No idea. send_101a(NPCid, PartyPos) -> send(<< (header(16#101a))/binary, NPCid:16/little-unsigned-integer, PartyPos:16/little-unsigned-integer, 16#ffffffff:32 >>). %% @todo Totally unknown. send_1020() -> send(header(16#1020)). %% @doc Update HP in the party members information on the left. %% @todo Figure out more of it. send_1022(HP) -> 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 >>). %% @todo Boss related command. 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 >>). %% @todo Boss related command. send_1113(Data) -> send(<< (header(16#1113))/binary, Data/binary >>). %% @todo Figure out what this packet does. Sane values for counter and missions for now. send_1202() -> 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. send_1204() -> send(<< (header(16#1204))/binary, 0:32, 16#20000000:32, 0:256 >>). %% @doc Object events response? %% @todo Not sure what Value does exactly. It's either 0 or 1. send_1205(EventID, BlockID, Value) -> 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. send_1206() -> 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. 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 >>, 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 Apparently A would be TargetID/ffffffff, B would be the player LID, C would be the object type? D still completely unknown. 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 >>). %% @doc Make the client load the quest previously sent. send_1212() -> send(<< (header(16#1212))/binary, 0:19200 >>). %% @todo Not sure. Related to keys. send_1213(A, B) -> send(<< (header(16#1213))/binary, A:32/little-unsigned-integer, B:32/little-unsigned-integer >>). %% @todo Related to boss gates. send_1215(A, B) -> send(<< (header(16#1215))/binary, A:32/little-unsigned-integer, 0:16, B:16/little-unsigned-integer >>). %% @todo Not sure yet. Value is probably a TargetID. Used in Airboard Rally. Replying with the same value starts the race. send_1216(Value) -> GID = get(gid), send(<< 16#12160300:32, 0:32, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, Value:32/little-unsigned-integer >>). %% @todo Figure out this room packet. send_1309() -> {ok, << _Size:32, Packet/bits >>} = file:read_file("p/packet1309.bin"), send(Packet). %% @todo Figure out this room packet. send_1332() -> {ok, << _Size:32, Packet/bits >>} = file:read_file("p/packet1332.bin"), send(Packet). %% @doc Send the player's partner card. %% @todo Find out the remaining values. send_1500(User) -> GID = User#egs_user_model.id, #characters{slot=Slot, name=Name, race=Race, gender=Gender, class=Class} = User#egs_user_model.character, RaceBin = psu_characters:race_atom_to_binary(Race), GenderBin = psu_characters:gender_atom_to_binary(Gender), ClassBin = psu_characters:class_atom_to_binary(Class), send(<< 16#15000300:32, 16#ffff:16, 0:144, 16#00011300:32, GID:32/little, 0:64, Name/binary, RaceBin:8, GenderBin:8, ClassBin:8, 0:40, GID:32/little, 0:3040, 16#010401:24, Slot:8, 0:64 >>). %% @todo Send an empty partner card list. send_1501() -> GID = get(gid), send(<< 16#15010300:32, 16#ffff:16, 0:144, 16#00011300:32, GID:32/little, 0:96 >>). %% @todo Send an empty blacklist. send_1512() -> GID = get(gid), send(<< 16#15120300:32, 16#ffff:16, 0:144, 16#00011300:32, GID:32/little, 0:46144 >>). %% @todo NPC related packet, sent when there's an NPC in the area. send_1601(PartyPos) -> {ok, << _:32, Bin/bits >>} = file:read_file("p/packet1601.bin"), send(<< (header(16#1601))/binary, PartyPos:32/little, Bin/binary >>). %% @doc Send the player's NPC and PM information. %% @todo The value 4 is the card priority. Find what 3 is. When sending, the first 0 is an unknown value. send_1602() -> NbNPC = lists:sum([1 || {_NPCid, Data} <- ?NPC, Data#psu_npc.has_card =:= true]), Bin = iolist_to_binary([<< NPCid:8, 0, 4, 0, 3, 0:24 >> || {NPCid, Data} <- ?NPC, Data#psu_npc.has_card =:= true]), MiddlePaddingSize = 8 * (344 - byte_size(Bin)), PMName = "My PM", UCS2PMName = << << X:8, 0:8 >> || X <- PMName >>, EndPaddingSize = 8 * (64 - byte_size(UCS2PMName)), GID = get(gid), send(<< 16#16020300:32, 16#ffff:16, 0:144, 16#00011300:32, GID:32/little, 0:96, Bin/binary, 0:MiddlePaddingSize, NbNPC, 0:24, UCS2PMName/binary, 0:EndPaddingSize, 0:32 >>). %% @doc Party information. %% @todo Handle existing parties. send_1706(CharName) -> 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 >>). %% @doc Party settings. Item distribution is random for now. %% @todo Handle correctly. send_170a() -> send(<< (header(16#170a))/binary, 16#01010c08:32 >>). %% @todo Find what the heck this packet is. send_170c() -> {ok, File} = file:read_file("p/packet170c.bin"), send(<< (header(16#170c))/binary, File/binary >>). %% @doc Send the background to use for the counter. %% @todo Background has more info past the first byte. send_1711(Background) -> send(<< (header(16#1711))/binary, Background:32/little-unsigned-integer >>). %% @doc Unknown dialog-related handler. %% @todo Everything! send_1a02(A, B, C, D, E) -> 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 >>). %% @doc Lumilass handler. Possibly more. %% @todo Figure out how Lumilass work exactly. The 4 bytes before the file may vary. send_1a03() -> {ok, File} = file:read_file("p/lumilassA.bin"), send(<< (header(16#1a03))/binary, 0:32, File/binary >>). %% @doc PP cube handler. %% @todo The 4 bytes before the file may vary. Everything past that is the same. Figure things out. send_1a04() -> {ok, File} = file:read_file("p/ppcube.bin"), send(<< (header(16#1a04))/binary, 0:32, File/binary >>). %% @doc Types menu handler. %% @todo Handle correctly. send_1a07() -> 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 >>). %% @doc Log message to the console. log(Message) -> io:format("game (~p): ~s~n", [get(gid), Message]). log(Message, Format) -> FormattedMessage = io_lib:format(Message, Format), log(FormattedMessage).