2010-07-18 20:08:11 +08:00
%% @author Lo<4C> c Hoguin <essen@dev-extend.eu>
%% @copyright 2010 Lo<4C> 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 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 General Public License for more details.
%%
%% You should have received a copy of the GNU General Public License
%% along with EGS. If not, see <http://www.gnu.org/licenses/>.
- module ( psu_game ) .
2010-07-18 21:41:26 +08:00
- export ( [ start_link / 1 , cleanup / 1 ] ) . %% External.
2010-08-23 04:34:44 +08:00
- export ( [ listen / 2 , accept / 2 , process_init / 2 , process / 0 , char_select / 0 , loop / 1 ] ) . %% Internal.
2010-05-13 12:34:04 +08:00
- include ( " include/records.hrl " ) .
- include ( " include/maps.hrl " ) .
2010-07-24 03:46:03 +08:00
- include ( " include/missions.hrl " ) .
2010-08-17 00:31:01 +08:00
- include ( " include/psu/npc.hrl " ) .
2010-05-13 12:34:04 +08:00
2010-07-31 19:55:54 +08:00
- define ( OPTIONS , [ binary , { active , false } , { reuseaddr , true } , { certfile , " priv/ssl/servercert.pem " } , { keyfile , " priv/ssl/serverkey.pem " } , { password , " alpha " } ] ) .
2010-05-13 12:34:04 +08:00
2010-07-18 20:08:11 +08:00
%% @spec start_link(Port) -> {ok,Pid::pid()}
%% @doc Start the game server.
start_link ( Port ) - >
2010-07-18 21:41:26 +08:00
{ ok , MPid } = egs_exit_mon : start_link ( { ? MODULE , cleanup } ) ,
LPid = spawn ( ? MODULE , listen , [ Port , MPid ] ) ,
2010-07-18 07:44:19 +08:00
{ ok , LPid } .
2010-05-13 12:34:04 +08:00
2010-07-18 21:41:26 +08:00
%% @spec cleanup(Pid) -> ok
%% @doc Cleanup the data associated with the failing process.
2010-07-24 03:46:03 +08:00
%% @todo Cleanup the instance process if there's nobody in it anymore.
2010-07-18 21:41:26 +08:00
cleanup ( Pid ) - >
2010-07-20 16:15:19 +08:00
case egs_user_model : read ( { pid , Pid } ) of
{ ok , User } - >
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 .
2010-06-14 01:05:28 +08:00
%% @doc Listen for connections.
2010-07-18 21:41:26 +08:00
listen ( Port , MPid ) - >
2010-07-31 19:55:54 +08:00
error_logger : info_report ( io_lib : format ( " psu_game listening on port ~b " , [ Port ] ) ) ,
2010-07-18 20:08:11 +08:00
{ ok , LSocket } = ssl : listen ( Port , ? OPTIONS ) ,
2010-07-18 21:41:26 +08:00
? MODULE : accept ( LSocket , MPid ) .
2010-05-13 12:34:04 +08:00
%% @doc Accept connections.
2010-07-18 21:41:26 +08:00
accept ( LSocket , MPid ) - >
2010-05-13 12:34:04 +08:00
case ssl : transport_accept ( LSocket , 5000 ) of
{ ok , CSocket } - >
2010-07-31 19:55:54 +08:00
case ssl : ssl_accept ( CSocket , 5000 ) of
ok - >
Pid = spawn ( ? MODULE , process_init , [ CSocket , MPid ] ) ,
ssl : controlling_process ( CSocket , Pid ) ;
{ error , _ Reason } - >
reload
end ;
2010-05-20 06:10:02 +08:00
_ - >
2010-05-13 12:34:04 +08:00
reload
end ,
2010-07-18 21:41:26 +08:00
? MODULE : accept ( LSocket , MPid ) .
2010-05-13 12:34:04 +08:00
2010-06-28 07:33:31 +08:00
%% @doc Initialize the client process by saving the socket to the process dictionary.
2010-07-18 21:41:26 +08:00
process_init ( CSocket , MPid ) - >
link ( MPid ) ,
2010-06-28 07:33:31 +08:00
put ( socket , CSocket ) ,
send_0202 ( ) ,
2010-07-18 23:54:28 +08:00
timer : send_interval ( 5000 , { psu_keepalive } ) ,
2010-06-28 07:33:31 +08:00
process ( ) .
2010-05-13 12:34:04 +08:00
%% @doc Process the new connections.
%% Send an hello packet, authenticate the user and send him to character select.
2010-06-28 07:33:31 +08:00
process ( ) - >
2010-08-19 07:53:18 +08:00
case psu_proto : packet_recv ( get ( socket ) , 5000 ) of
2010-06-04 06:09:08 +08:00
{ ok , Orig } - >
2010-08-19 07:53:18 +08:00
{ command , Command , _ , Data } = psu_proto : packet_parse ( Orig ) ,
2010-06-28 07:33:31 +08:00
process_handle ( Command , Data ) ;
2010-05-13 12:34:04 +08:00
{ error , timeout } - >
reload ,
2010-06-28 07:33:31 +08:00
? MODULE : process ( ) ;
2010-05-13 12:34:04 +08:00
{ error , closed } - >
2010-06-04 05:36:34 +08:00
closed
2010-05-13 12:34:04 +08:00
end .
2010-06-28 07:33:31 +08:00
%% @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 > > ) - >
CSocket = get ( socket ) ,
2010-07-19 08:11:31 +08:00
case egs_user_model : read ( GID ) of
{ error , badarg } - >
2010-06-28 07:33:31 +08:00
log ( " can't find user, closing " ) ,
2010-05-15 22:12:00 +08:00
ssl : close ( CSocket ) ;
2010-07-19 08:11:31 +08:00
{ ok , User } - >
case User #egs_user_model.state of
{ wait_for_authentication , Auth } - >
2010-06-30 18:11:50 +08:00
put ( gid , GID ) ,
2010-06-28 07:33:31 +08:00
log ( " auth success " ) ,
2010-08-01 03:32:08 +08:00
LID = 1 + mnesia : dirty_update_counter ( counters , lobby , 1 ) rem 1023 ,
2010-05-21 05:51:13 +08:00
Time = calendar : datetime_to_gregorian_seconds ( calendar : universal_time ( ) ) ,
2010-07-19 08:11:31 +08:00
egs_user_model : write ( #egs_user_model { id = GID , pid = self ( ) , socket = CSocket , state = authenticated , time = Time , folder = User #egs_user_model.folder , lid = LID } ) ,
2010-06-28 07:33:31 +08:00
send_0d05 ( ) ,
? MODULE : char_select ( ) ;
2010-05-15 22:12:00 +08:00
_ - >
2010-06-28 07:33:31 +08:00
log ( " quit, auth failed " ) ,
2010-07-19 08:11:31 +08:00
egs_user_model : delete ( GID ) ,
2010-05-15 22:12:00 +08:00
ssl : close ( CSocket )
end
2010-05-13 12:34:04 +08:00
end ;
%% @doc Unknown command handler. Do nothing.
2010-06-28 07:33:31 +08:00
process_handle ( Command , _ ) - >
log ( " (process) dismissed packet ~4.16.0b " , [ Command ] ) ,
? MODULE : process ( ) .
2010-05-13 12:34:04 +08:00
%% @doc Character selection screen loop.
2010-06-28 07:33:31 +08:00
char_select ( ) - >
2010-08-19 07:53:18 +08:00
case psu_proto : packet_recv ( get ( socket ) , 5000 ) of
2010-06-04 06:09:08 +08:00
{ ok , Orig } - >
2010-08-19 07:53:18 +08:00
{ command , Command , _ , Data } = psu_proto : packet_parse ( Orig ) ,
2010-06-28 07:33:31 +08:00
char_select_handle ( Command , Data ) ;
2010-05-13 12:34:04 +08:00
{ error , timeout } - >
2010-08-19 07:53:18 +08:00
psu_proto : send_keepalive ( get ( socket ) ) ,
2010-05-13 12:34:04 +08:00
reload ,
2010-06-28 07:33:31 +08:00
? MODULE : char_select ( ) ;
2010-05-13 12:34:04 +08:00
{ error , closed } - >
2010-07-20 16:15:19 +08:00
closed %% exit
2010-05-13 12:34:04 +08:00
end .
%% @doc Character selection handler.
2010-06-28 07:33:31 +08:00
char_select_handle ( 16#020b , < < Number : 32 / little - unsigned - integer , _ / bits > > ) - >
log ( " selected character ~b " , [ Number ] ) ,
char_select_load ( Number ) ;
2010-05-13 12:34:04 +08:00
%% @doc Character creation handler.
2010-07-20 02:45:09 +08:00
%% @todo Reenable appearance validation whenever things go live.
2010-06-28 07:33:31 +08:00
char_select_handle ( 16#0d02 , < < Number : 32 / little - unsigned - integer , Char / bits > > ) - >
log ( " character creation " ) ,
2010-07-20 02:45:09 +08:00
%% check for valid character appearance
%~ << _Name:512, RaceID:8, GenderID:8, _TypeID:8, AppearanceBin:776/bits, _/bits >> = Char,
%~ 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
2010-07-19 08:11:31 +08:00
{ 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 , Number ] ) , Char ) ,
file : write_file ( io_lib : format ( " save/ ~s / ~b -character.options " , [ User #egs_user_model.folder , Number ] ) , < < 0 : 128 , 4 , 0 : 56 > > ) , % default 0 to everything except brightness 4
2010-06-28 07:33:31 +08:00
char_select_load ( Number ) ;
2010-05-13 12:34:04 +08:00
%% @doc Character selection screen request.
2010-06-28 07:33:31 +08:00
char_select_handle ( 16#0d06 , _ ) - >
2010-07-19 08:11:31 +08:00
{ 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 ) ) ,
2010-06-28 07:33:31 +08:00
? MODULE : char_select ( ) ;
2010-05-13 12:34:04 +08:00
2010-06-04 05:45:06 +08:00
%% @doc Silently ignore packet 0818. Gives CPU/GPU information.
2010-06-28 07:33:31 +08:00
char_select_handle ( 16#0818 , _ ) - >
? MODULE : char_select ( ) ;
2010-06-04 05:45:06 +08:00
2010-05-13 12:34:04 +08:00
%% @doc Unknown command handler. Do nothing.
2010-06-28 07:33:31 +08:00
char_select_handle ( Command , _ ) - >
log ( " (char_select) dismissed packet ~4.16.0b " , [ Command ] ) ,
? MODULE : char_select ( ) .
2010-05-13 12:34:04 +08:00
2010-06-08 23:21:25 +08:00
%% @doc Load the selected character in the start lobby and start the main game's loop.
2010-07-10 00:54:37 +08:00
%% The default entry point currently is 4th floor, Linear Line counter.
2010-06-28 07:33:31 +08:00
char_select_load ( Number ) - >
2010-07-19 08:11:31 +08:00
{ ok , OldUser } = egs_user_model : read ( get ( gid ) ) ,
[ { status , 1 } , { char , CharBin } , { options , OptionsBin } ] = data_load ( OldUser #egs_user_model.folder , Number ) ,
2010-07-02 03:26:26 +08:00
< < 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
2010-08-21 22:52:09 +08:00
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 } ,
2010-07-19 08:11:31 +08:00
egs_user_model : write ( User ) ,
2010-07-02 03:26:26 +08:00
char_load ( User ) ,
2010-06-28 07:33:31 +08:00
send_021b ( ) ,
2010-07-16 22:25:31 +08:00
area_load ( 1100000 , 0 , 4 , 5 ) ,
2010-06-28 07:33:31 +08:00
ssl : setopts ( get ( socket ) , [ { active , true } ] ) ,
? MODULE : loop ( < < > > ) .
2010-06-08 23:21:25 +08:00
2010-05-13 12:34:04 +08:00
%% @doc Load the given character's data.
2010-06-08 23:04:54 +08:00
data_load ( Folder , Number ) - >
2010-05-14 02:45:17 +08:00
Filename = io_lib : format ( " save/ ~s / ~b -character " , [ Folder , Number ] ) ,
2010-05-13 12:34:04 +08:00
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 .
2010-06-08 23:21:25 +08:00
%% @doc Load and send the character information to the client.
2010-07-02 03:26:26 +08:00
char_load ( User ) - >
send_0d01 ( User ) ,
2010-06-04 21:01:04 +08:00
% 0246
2010-06-28 07:33:31 +08:00
send_0a0a ( ) ,
send_1006 ( 5 ) ,
2010-07-19 08:11:31 +08:00
send_1005 ( ( User #egs_user_model.character ) #characters.name ) ,
2010-06-28 07:33:31 +08:00
send_1006 ( 12 ) ,
2010-07-03 02:06:13 +08:00
send_0210 ( ) ,
2010-06-28 07:33:31 +08:00
send_0222 ( ) ,
2010-07-02 03:26:26 +08:00
send_1500 ( User ) ,
2010-06-28 07:33:31 +08:00
send_1501 ( ) ,
send_1512 ( ) ,
2010-06-04 23:02:13 +08:00
% 0303
2010-06-28 07:33:31 +08:00
send_1602 ( ) .
2010-05-13 12:34:04 +08:00
2010-06-17 10:03:48 +08:00
%% @doc Return the current season information.
area_get_season ( QuestID ) - >
{ { _ , Month , Day } , _ } = calendar : universal_time ( ) ,
2010-07-10 00:54:37 +08:00
[ IsSeasonal , SeasonID , SeasonQuestIDs ] = if
2010-06-17 10:03:48 +08:00
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 ;
2010-06-17 18:35:47 +08:00
Month =:= 4 , Day > = 4 , Day =< 24 - > ? SEASON_EASTER ;
Month =:= 4 , Day > = 25 - > ? SEASON_SPRING ;
Month =:= 5 , Day =< 8 - > ? SEASON_SPRING ;
2010-06-17 10:03:48 +08:00
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
2010-07-10 00:54:37 +08:00
true - > [ { status , IsSeasonal } , { season , SeasonID } ] ;
2010-06-17 10:03:48 +08:00
false - > [ { status , 0 } , { season , 255 } ]
end ;
true - >
[ { status , 0 } , { season , 255 } ]
end .
2010-05-13 12:34:04 +08:00
%% @doc Load the given map as a standard lobby.
2010-06-28 07:33:31 +08:00
area_load ( QuestID , ZoneID , MapID , EntryID ) - >
2010-07-19 08:11:31 +08:00
{ ok , OldUser } = egs_user_model : read ( get ( gid ) ) ,
2010-07-24 18:51:51 +08:00
[ { type , AreaType } , { file , QuestFile } | MissionInfo ] = proplists : get_value ( QuestID , ? QUESTS , [ { type , undefined } , { file , undefined } ] ) ,
[ IsStart , RealZoneID , RealMapID , RealEntryID , NbSetsInQuest ] = case AreaType of
2010-06-10 00:40:19 +08:00
mission - >
if ZoneID =:= 65535 - >
2010-07-24 18:51:51 +08:00
[ { start , [ TmpZoneID , TmpMapID , TmpEntryID ] } , { sets , TmpNbSets } ] = MissionInfo ,
[ true , TmpZoneID , TmpMapID , TmpEntryID , TmpNbSets ] ;
true - > [ false , ZoneID , MapID , EntryID , ignored ]
2010-06-10 00:40:19 +08:00
end ;
2010-06-13 03:26:37 +08:00
myroom - >
if ZoneID =:= 0 - >
2010-07-24 18:51:51 +08:00
[ false , 0 , 423 , EntryID , ignored ] ;
true - > [ false , ZoneID , MapID , EntryID , ignored ]
2010-06-13 03:26:37 +08:00
end ;
2010-06-10 00:40:19 +08:00
_ - >
2010-07-24 18:51:51 +08:00
[ false , ZoneID , MapID , EntryID , ignored ]
2010-06-10 00:40:19 +08:00
end ,
2010-07-24 18:51:51 +08:00
[ { file , ZoneFile } | ZoneSetInfo ] = proplists : get_value ( [ QuestID , RealZoneID ] , ? ZONES , [ { file , undefined } ] ) ,
NbSetsInZone = case ZoneSetInfo of [ ] - > 1 ; [ { sets , TmpNbSetsInZone } ] - > TmpNbSetsInZone end ,
2010-06-13 03:26:37 +08:00
if AreaType =:= myroom - >
AreaName = " Your Room " ;
true - >
[ { name , AreaName } ] = proplists : get_value ( [ QuestID , RealMapID ] , ? MAPS , [ { name , " dammy " } ] )
end ,
2010-07-24 18:51:51 +08:00
{ InstancePid , SetID } = if IsStart =:= true - > % initialize the mission
2010-07-24 03:46:03 +08:00
Zones = proplists : get_value ( QuestID , ? MISSIONS ) ,
{ ok , RetPid } = psu_instance : start_link ( Zones ) ,
2010-07-24 18:51:51 +08:00
RetSetID = crypto : rand_uniform ( 0 , NbSetsInQuest ) ,
{ RetPid , RetSetID } ;
true - > { OldUser #egs_user_model.instancepid , OldUser #egs_user_model.setid }
2010-07-10 00:54:37 +08:00
end ,
2010-08-22 00:42:23 +08:00
User = OldUser #egs_user_model { instancepid = InstancePid , areatype = AreaType , area = { psu_area , QuestID , RealZoneID , RealMapID } , entryid = RealEntryID } ,
2010-07-24 03:46:03 +08:00
egs_user_model : write ( User ) ,
2010-07-24 18:51:51 +08:00
RealSetID = if SetID > NbSetsInZone - 1 - > NbSetsInZone - 1 ; true - > SetID end ,
area_load ( AreaType , IsStart , RealSetID , OldUser , User , QuestFile , ZoneFile , AreaName ) .
2010-06-10 00:40:19 +08:00
2010-07-10 00:54:37 +08:00
area_load ( AreaType , IsStart , SetID , OldUser , User , QuestFile , ZoneFile , AreaName ) - >
2010-07-19 08:11:31 +08:00
#psu_area { questid = OldQuestID , zoneid = OldZoneID } = OldUser #egs_user_model.area ,
2010-07-31 21:20:25 +08:00
#psu_area { questid = QuestID , zoneid = ZoneID , mapid = _ MapID } = User #egs_user_model.area ,
2010-07-19 08:11:31 +08:00
QuestChange = if OldQuestID / = QuestID , QuestFile / = undefined - > true ; true - > false end ,
2010-06-13 03:26:37 +08:00
if ZoneFile =:= undefined - >
ZoneChange = false ;
true - >
2010-07-19 08:11:31 +08:00
ZoneChange = if OldQuestID =:= QuestID , OldZoneID =:= ZoneID - > false ; true - > true end
2010-06-13 03:26:37 +08:00
end ,
2010-07-19 08:11:31 +08:00
[ { status , IsSeasonal } , { season , SeasonID } ] = area_get_season ( QuestID ) ,
2010-06-14 01:05:28 +08:00
% broadcast spawn and unspawn to other people
2010-07-19 08:11:31 +08:00
{ 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 ) ,
2010-06-14 01:05:28 +08:00
if AreaType =:= lobby - >
2010-07-19 08:11:31 +08:00
lists : foreach ( fun ( Other ) - > Other #egs_user_model.pid ! { psu_player_spawn , User } end , SpawnList ) ;
2010-06-14 01:05:28 +08:00
true - > ignore
end ,
% load area
if QuestChange =:= true - >
% reload the character if entering or leaving the room quest
2010-07-19 08:11:31 +08:00
if OldQuestID =:= 1120000 ; QuestID =:= 1120000 - >
2010-07-02 03:26:26 +08:00
char_load ( User ) ;
2010-06-14 01:05:28 +08:00
true - > ignore
end ,
% load new quest
2010-07-19 08:11:31 +08:00
send_0c00 ( QuestID ) ,
2010-06-28 07:33:31 +08:00
send_020e ( QuestFile ) ;
2010-06-14 01:05:28 +08:00
true - > ignore
end ,
if IsStart =:= true - >
2010-06-28 07:33:31 +08:00
send_0215 ( 16#ffffffff ) ;
2010-06-14 01:05:28 +08:00
true - > ignore
end ,
if ZoneChange =:= true - >
% load new zone
2010-06-28 07:33:31 +08:00
send_0a05 ( ) ,
2010-06-14 01:05:28 +08:00
if AreaType =:= lobby - >
2010-07-11 04:18:28 +08:00
send_0111 ( 6 , 0 ) ;
2010-06-14 01:05:28 +08:00
true - > ignore
end ,
2010-08-17 00:31:01 +08:00
send_010d ( User #egs_user_model { lid = 0 } ) ,
2010-06-28 07:33:31 +08:00
send_0200 ( AreaType ) ,
2010-07-10 00:54:37 +08:00
send_020f ( ZoneFile , SetID , SeasonID ) ;
2010-06-14 01:05:28 +08:00
true - > ignore
end ,
2010-07-19 08:11:31 +08:00
send_0205 ( ZoneID , ( User #egs_user_model.area ) #psu_area.mapid , User #egs_user_model.entryid , IsSeasonal ) ,
send_100e ( QuestID , ZoneID , ( User #egs_user_model.area ) #psu_area.mapid , AreaName , 16#ffffffff ) ,
2010-06-14 01:05:28 +08:00
if AreaType =:= mission - >
2010-06-28 07:33:31 +08:00
send_0215 ( 0 ) ,
2010-06-14 01:05:28 +08:00
if IsStart =:= true - >
2010-06-28 07:33:31 +08:00
send_0215 ( 0 ) ,
send_0c09 ( ) ;
2010-06-14 01:05:28 +08:00
true - > ignore
end ;
true - >
2010-06-28 07:33:31 +08:00
send_020c ( )
2010-06-14 01:05:28 +08:00
end ,
case AreaType of
myroom - >
2010-08-01 00:28:31 +08:00
send_1332 ( ) ,
2010-06-28 07:33:31 +08:00
send_1202 ( ) ,
send_1204 ( ) ,
send_1206 ( ) ;
2010-06-14 01:05:28 +08:00
mission - >
2010-06-28 07:33:31 +08:00
send_1202 ( ) ,
send_1204 ( ) ,
send_1206 ( ) ,
send_1207 ( ) ;
2010-06-14 01:05:28 +08:00
_ - > ignore
end ,
if AreaType / = spaceport - >
2010-06-28 07:33:31 +08:00
send_1212 ( ) ;
2010-06-14 01:05:28 +08:00
true - > ignore
end ,
if AreaType =:= myroom - >
2010-08-01 00:28:31 +08:00
send_1309 ( ) ;
2010-06-14 01:05:28 +08:00
true - > ignore
end ,
2010-08-17 00:31:01 +08:00
send_0201 ( User #egs_user_model { lid = 0 } ) ,
2010-06-14 01:05:28 +08:00
if ZoneChange =:= true - >
2010-06-28 07:33:31 +08:00
send_0a06 ( ) ;
2010-06-14 01:05:28 +08:00
true - > ignore
end ,
2010-07-19 08:11:31 +08:00
send_0233 ( SpawnList ) ,
2010-06-28 07:33:31 +08:00
send_0208 ( ) ,
2010-08-17 00:31:01 +08:00
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 ) ,
send_010d ( NPCUser ) ,
send_0201 ( NPCUser ) ,
send_0215 ( 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 ( ) ,
send_1016 ( PartyPos ) ,
npc_load ( Leader , NPCList ) .
2010-05-30 16:34:55 +08:00
2010-05-13 12:34:04 +08:00
%% @doc Game's main loop.
2010-07-18 23:54:28 +08:00
%% @todo We probably don't want to send a keepalive packet unnecessarily.
2010-06-28 07:33:31 +08:00
loop ( SoFar ) - >
2010-05-13 12:34:04 +08:00
receive
2010-06-04 06:09:08 +08:00
{ psu_broadcast , Orig } - >
< < A : 64 / bits , _ : 32 , B : 96 / bits , _ : 64 , C / bits > > = Orig ,
2010-06-28 07:33:31 +08:00
GID = get ( gid ) ,
send ( < < A / binary , 16#00011300 : 32 , B / binary , 16#00011300 : 32 , GID : 32 / little - unsigned - integer , C / binary > > ) ,
? MODULE : loop ( SoFar ) ;
2010-08-17 00:31:01 +08:00
{ psu_chat , ChatTypeID , ChatGID , ChatName , ChatModifiers , ChatMessage } - >
send_0304 ( ChatTypeID , ChatGID , ChatName , ChatModifiers , ChatMessage ) ,
2010-06-28 07:33:31 +08:00
? MODULE : loop ( SoFar ) ;
2010-05-20 17:35:25 +08:00
{ psu_keepalive } - >
2010-08-19 07:53:18 +08:00
psu_proto : send_keepalive ( get ( socket ) ) ,
2010-06-28 07:33:31 +08:00
? MODULE : loop ( SoFar ) ;
2010-06-28 18:10:31 +08:00
{ psu_player_spawn , _ Spawn } - >
% Should be something along the lines of 203 201 204 or something.
2010-07-19 08:11:31 +08:00
{ ok , User } = egs_user_model : read ( get ( gid ) ) ,
{ ok , SpawnList } = egs_user_model : select ( { neighbors , User } ) ,
send_0233 ( SpawnList ) ,
2010-06-28 07:33:31 +08:00
? MODULE : loop ( SoFar ) ;
2010-06-02 05:40:32 +08:00
{ psu_player_unspawn , Spawn } - >
2010-07-19 08:11:31 +08:00
send_0204 ( Spawn #egs_user_model.id , Spawn #egs_user_model.lid , 5 ) ,
2010-06-28 07:33:31 +08:00
? MODULE : loop ( SoFar ) ;
2010-06-30 20:02:52 +08:00
{ psu_warp , QuestID , ZoneID , MapID , EntryID } - >
area_load ( QuestID , ZoneID , MapID , EntryID ) ,
? MODULE : loop ( SoFar ) ;
2010-05-13 12:34:04 +08:00
{ ssl , _ , Data } - >
2010-08-19 07:53:18 +08:00
{ Packets , Rest } = psu_proto : packet_split ( < < SoFar / bits , Data / bits > > ) ,
2010-06-28 07:33:31 +08:00
[ dispatch ( Orig ) | | Orig < - Packets ] ,
? MODULE : loop ( Rest ) ;
2010-05-13 12:34:04 +08:00
{ ssl_closed , _ } - >
2010-07-20 16:15:19 +08:00
ssl_closed ; %% exit
2010-05-13 12:34:04 +08:00
{ ssl_error , _ , _ } - >
2010-07-20 16:15:19 +08:00
ssl_error ; %% exit
2010-05-13 12:34:04 +08:00
_ - >
2010-06-28 07:33:31 +08:00
? MODULE : loop ( SoFar )
2010-05-13 12:34:04 +08:00
after 1000 - >
reload ,
2010-06-28 07:33:31 +08:00
? MODULE : loop ( SoFar )
2010-05-13 12:34:04 +08:00
end .
%% @doc Dispatch the command to the right handler.
2010-06-28 07:33:31 +08:00
dispatch ( Orig ) - >
2010-08-20 05:43:19 +08:00
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 )
2010-05-20 13:47:28 +08:00
end .
%% @doc Position change broadcast handler. Save the position and then dispatch it.
2010-06-28 07:33:31 +08:00
broadcast ( 16#0503 , Orig ) - >
2010-07-17 00:49:15 +08:00
< < _ : 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 ,
2010-07-16 22:25:31 +08:00
FloatDir = Dir / 46603 . 375 ,
2010-07-19 08:11:31 +08:00
{ 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 ) ,
2010-06-28 07:33:31 +08:00
broadcast ( default , Orig ) ;
2010-05-20 13:47:28 +08:00
2010-05-26 16:10:14 +08:00
%% @doc Stand still broadcast handler. Save the position and then dispatch it.
2010-06-28 07:33:31 +08:00
broadcast ( 16#0514 , Orig ) - >
2010-07-16 22:25:31 +08:00
< < _ : 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 ,
2010-06-06 08:28:35 +08:00
MapID : 32 / little - unsigned - integer , EntryID : 32 / little - unsigned - integer , _ / bits > > = Orig ,
2010-07-16 22:25:31 +08:00
FloatDir = Dir / 46603 . 375 ,
2010-07-19 08:11:31 +08:00
{ 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 ) ,
2010-06-28 07:33:31 +08:00
broadcast ( default , Orig ) ;
2010-05-26 16:10:14 +08:00
2010-05-26 05:09:08 +08:00
%% @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.
2010-05-26 11:35:31 +08:00
%% Only a handful of commands are allowed to broadcast. An user tampering with it would get disconnected instantly.
2010-05-20 13:47:28 +08:00
%% @todo Don't query the user data everytime! Keep an User instead of a GID probably.
2010-06-28 07:33:31 +08:00
broadcast ( Command , Orig )
2010-05-26 05:09:08 +08:00
when Command =:= 16#0101 ;
Command =:= 16#0102 ;
Command =:= 16#0104 ;
Command =:= 16#0107 ;
Command =:= 16#010f ;
Command =:= 16#050f ;
Command =:= default - >
2010-06-04 06:09:08 +08:00
< < _ : 32 , A : 64 / bits , _ : 64 , B : 192 / bits , _ : 64 , C / bits > > = Orig ,
2010-06-28 07:33:31 +08:00
GID = get ( gid ) ,
2010-07-19 08:11:31 +08:00
case egs_user_model : read ( GID ) of
{ error , _ Reason } - >
2010-05-20 13:47:28 +08:00
ignore ;
2010-07-19 08:11:31 +08:00
{ ok , Self } - >
LID = Self #egs_user_model.lid ,
2010-06-04 06:09:08 +08:00
Packet = < < A / binary , 16#00011300 : 32 , GID : 32 / little - unsigned - integer , B / binary ,
2010-05-20 13:47:28 +08:00
GID : 32 / little - unsigned - integer , LID : 32 / little - unsigned - integer , C / binary > > ,
2010-07-19 08:11:31 +08:00
{ ok , SpawnList } = egs_user_model : select ( { neighbors , Self } ) ,
lists : foreach ( fun ( User ) - > User #egs_user_model.pid ! { psu_broadcast , Packet } end , SpawnList )
2010-05-20 13:47:28 +08:00
end .
2010-08-26 20:51:24 +08:00
%% @doc Trigger many events.
events ( Events ) - >
[ event ( Event ) | | Event < - Events ] ,
ok .
2010-08-21 21:40:17 +08:00
%% @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 ) ;
2010-08-23 20:45:47 +08:00
%% @doc Chat broadcast handler. Dispatch the message to everyone (for now).
%% Disregard the name sent by the server. Use the name saved in memory instead, to prevent client-side editing.
%% @todo Only broadcast to people in the same map.
%% @todo In the case of NPC characters, when FromTypeID is 00001d00, check that the NPC is in the party and broadcast only to the party (probably).
%% @todo When the game doesn't find an NPC (probably) and forces it to talk like in the tutorial mission it seems FromTypeID, FromGID and Name are all 0.
%% @todo Make sure modifiers have correct values.
event ( { chat , _ FromTypeID , FromGID , _ FromName , Modifiers , ChatMsg } ) - >
UserGID = get ( gid ) ,
[ BcastTypeID , BcastGID , BcastName ] = case FromGID of
0 - > %% @todo unknown; just fail for now.
log ( " event chat FromGID=0 " ) ,
ignore ; %% @todo This will trigger an error.
UserGID - > %% player chat: disregard whatever was sent except modifiers and message.
{ ok , User } = egs_user_model : read ( UserGID ) ,
[ 16#00001200 , User #egs_user_model.id , ( User #egs_user_model.character ) #characters.name ] ;
NPCGID - > %% npc chat: @todo Check that the player is the party leader and this npc is in his party.
{ ok , User } = egs_user_model : read ( NPCGID ) ,
[ 16#00001d00 , FromGID , ( User #egs_user_model.character ) #characters.name ]
end ,
%% log the message as ascii to the console
if FromGID =:= 0 - > ignore ; true - >
[ LogName | _ ] = re : split ( BcastName , " \\ 0 \\ 0 " , [ { return , binary } ] ) ,
[ TmpMessage | _ ] = re : split ( ChatMsg , " \\ 0 \\ 0 " , [ { return , binary } ] ) ,
LogMessage = re : replace ( TmpMessage , " \\ n " , " " , [ global , { return , binary } ] ) ,
log ( " chat from ~s : ~s " , [ [ re : replace ( LogName , " \\ 0 " , " " , [ global , { return , binary } ] ) ] , [ re : replace ( LogMessage , " \\ 0 " , " " , [ global , { return , binary } ] ) ] ] )
end ,
%% broadcast
{ ok , List } = egs_user_model : select ( all ) ,
lists : foreach ( fun ( X ) - > X #egs_user_model.pid ! { psu_chat , BcastTypeID , BcastGID , BcastName , Modifiers , ChatMsg } end , List ) ;
2010-08-21 22:52:09 +08:00
%% @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 } ,
2010-08-22 00:42:23 +08:00
User = OldUser #egs_user_model { areatype = counter , area = { psu_area , 16#7fffffff , 0 , 0 } , entryid = 0 , prev_area = FromArea , prev_entryid = FromEntryID } ,
2010-08-21 22:52:09 +08:00
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
send_0c00 ( 16#7fffffff ) ,
send_020e ( QuestFile ) ,
send_0a05 ( ) ,
send_010d ( User #egs_user_model { lid = 0 } ) ,
send_0200 ( mission ) ,
send_020f ( ZoneFile , 0 , 16#ff ) ,
send_0205 ( 0 , 0 , 0 , 0 ) ,
send_100e ( 16#7fffffff , 0 , 0 , AreaName , CounterID ) ,
send_0215 ( 0 ) ,
send_0215 ( 0 ) ,
send_020c ( ) ,
send_1202 ( ) ,
send_1204 ( ) ,
send_1206 ( ) ,
send_1207 ( ) ,
send_1212 ( ) ,
send_0201 ( User #egs_user_model { lid = 0 } ) ,
send_0a06 ( ) ,
send_0208 ( ) ,
send_0236 ( ) ;
2010-08-21 22:59:01 +08:00
%% @doc Leave mission counter handler.
event ( counter_leave ) - >
{ ok , User } = egs_user_model : read ( get ( gid ) ) ,
PrevArea = User #egs_user_model.prev_area ,
area_load ( PrevArea #psu_area.questid , PrevArea #psu_area.zoneid , PrevArea #psu_area.mapid , User #egs_user_model.prev_entryid ) ;
2010-08-22 00:39:15 +08:00
%% @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 ) ;
2010-08-23 05:00:43 +08:00
%% @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 ) ;
2010-08-23 05:08:01 +08:00
%% @todo Item distribution is always set to random for now.
event ( counter_party_options_request ) - >
send_170a ( ) ;
2010-08-21 23:42:31 +08:00
%% @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 ) ;
2010-08-22 00:16:21 +08:00
%% @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 ) ;
2010-08-26 20:41:35 +08:00
%% @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
2010-08-26 20:51:24 +08:00
events ( Events ) ;
2010-08-26 20:41:35 +08:00
_ - >
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 ;
2010-08-23 06:10:12 +08:00
%% @todo Send something other than just "dammy".
event ( { item_description_request , ItemID } ) - >
send_0a11 ( ItemID , " dammy " ) ;
2010-08-20 05:43:19 +08:00
%% @todo A and B are unknown.
2010-06-02 01:37:02 +08:00
%% 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.
2010-06-01 00:50:24 +08:00
%% @todo Others probably want to see that you changed your weapon.
2010-06-02 01:37:02 +08:00
%% @todo Apparently B is always ItemID+1. Not sure why.
%% @todo Currently use a separate file for the data sent for the weapons.
2010-08-20 05:43:19 +08:00
%% @todo TargetGID and TargetLID must be validated, they're either the player's or his NPC characters.
event ( { item_equip , ItemID , TargetGID , TargetLID , A , B } ) - >
2010-06-28 07:33:31 +08:00
GID = get ( gid ) ,
2010-06-20 04:04:42 +08:00
Category = case ItemID 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 ,
2010-08-20 05:43:19 +08:00
Filename = case ItemID 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 , 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 , ItemID , 1 , Category , A , B : 32 / little - unsigned - integer ,
File / binary > > ) ;
%% @todo A and B are unknown.
%% @see item_equip
event ( { item_unequip , ItemID , TargetGID , TargetLID , A , B } ) - >
GID = get ( gid ) ,
Category = case ItemID 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 ,
2010-08-21 05:22:24 +08:00
0 : 64 , TargetGID : 32 / little - unsigned - integer , TargetLID : 32 / little - unsigned - integer , ItemID , 2 , Category , A , B : 32 / little - unsigned - integer > > ) ;
2010-08-21 23:59:59 +08:00
%% @todo Just ignore the meseta price for now and send the player where he wanna be!
event ( lobby_transport_request ) - >
send_0c08 ( true ) ;
2010-08-23 01:15:08 +08:00
%% @todo Handle all different Lumilass.
event ( lumilass_options_request ) - >
send_1a03 ( ) ;
2010-08-22 00:08:32 +08:00
%% @todo Probably replenish the player HP when entering a non-mission area rather than when aborting the mission?
event ( mission_abort ) - >
send_1006 ( 11 ) ,
{ ok , User } = egs_user_model : read ( get ( gid ) ) ,
%% delete the mission
2010-08-23 05:03:19 +08:00
if User #egs_user_model.instancepid =:= undefined - > ignore ;
true - > psu_instance : stop ( User #egs_user_model.instancepid )
end ,
2010-08-22 00:08:32 +08:00
%% 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 - >
Area = User #egs_user_model.prev_area ,
area_load ( Area #psu_area.questid , Area #psu_area.zoneid , Area #psu_area.mapid , User #egs_user_model.prev_entryid ) ;
true - > ignore
end ;
2010-08-21 23:24:38 +08:00
%% @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_170c ( ) ,
send_1020 ( ) ,
send_1015 ( QuestID ) ,
send_0c02 ( ) ;
2010-08-21 23:08:21 +08:00
%% @todo Also happening a 1506 -> 1507? Only on first selection from menu.
%% @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.
%% @todo Check if party already exists.
{ ok , PartyPid } = psu_party : start_link ( GID ) ,
{ 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 } ,
SentNPCUser = NPCUser #egs_user_model { id = NPCid , character = SentNPCCharacter } ,
%% @todo send_022c(0, 2),
send_1004 ( npc_invite , SentNPCUser , PartyPos ) ,
send_101a ( NPCid , PartyPos ) ;
2010-08-23 02:33:16 +08:00
%% @todo Currently send the normal items shop for all shops, differentiate.
event ( { npc_shop_enter , ShopID } ) - >
log ( " npc shop enter ~p " , [ ShopID ] ) ,
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 > > ) ;
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 > > ) ;
2010-08-23 01:15:08 +08:00
%% @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 ;
2010-08-25 07:42:37 +08:00
%% @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 ;
2010-08-26 20:51:24 +08:00
%% @doc Server-side event.
event ( { object_box_destroy , ObjectID } ) - >
send_1213 ( ObjectID , 3 ) ;
2010-08-25 07:42:37 +08:00
%% @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 ) ;
2010-08-26 20:51:24 +08:00
%% @doc Server-side event.
event ( { object_event_trigger , BlockID , EventID } ) - >
send_1205 ( EventID , BlockID , 0 ) ;
2010-08-25 07:42:37 +08:00
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 ) ;
2010-08-23 01:15:08 +08:00
event ( player_type_availability_request ) - >
send_1a07 ( ) ;
2010-08-21 06:23:17 +08:00
%% @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 ) ) ,
Area = User #egs_user_model.prev_area ,
area_load ( Area #psu_area.questid , Area #psu_area.zoneid , Area #psu_area.mapid , User #egs_user_model.prev_entryid ) ;
event ( player_type_capabilities_request ) - >
send_0113 ( ) ;
2010-08-23 01:15:08 +08:00
event ( ppcube_request ) - >
send_1a04 ( ) ;
2010-08-21 05:22:24 +08:00
%% @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.
event ( { unicube_select , Selection , EntryID } ) - >
case Selection of
cancel - > ignore ;
16#ffffffff - >
log ( " uni selection (my room) " ) ,
send_0230 ( ) ,
% 0220
area_load ( 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 ) ) ,
if User #egs_user_model.partypid =:= undefined - >
ignore ;
true - >
%% @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)
psu_party : stop ( User #egs_user_model.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 ) ,
area_load ( Area #psu_area.questid , Area #psu_area.zoneid , Area #psu_area.mapid , EntryID )
end .
2010-08-20 05:43:19 +08:00
2010-08-23 01:15:08 +08:00
2010-08-20 05:43:19 +08:00
%% @doc Movement (non-broadcast) handler. Do nothing.
handle ( 16#0102 , _ ) - >
ignore ;
2010-06-01 00:50:24 +08:00
2010-06-19 00:33:13 +08:00
%% @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.
2010-06-28 07:33:31 +08:00
handle ( 16#0402 , Data ) - >
2010-06-28 01:59:16 +08:00
< < SpawnID : 32 / little - unsigned - integer , _ : 64 , Type : 32 / little - unsigned - integer , _ : 64 > > = Data ,
2010-06-19 00:33:13 +08:00
case Type of
7 - > % spawn cleared @todo 1201 sent back with same values apparently, but not always
2010-07-10 00:54:37 +08:00
log ( " cleared spawn ~b " , [ SpawnID ] ) ,
2010-07-19 08:11:31 +08:00
{ ok , User } = egs_user_model : read ( get ( gid ) ) ,
2010-07-24 03:46:03 +08:00
{ BlockID , EventID } = psu_instance : spawn_cleared_event ( User #egs_user_model.instancepid , ( User #egs_user_model.area ) #psu_area.zoneid , SpawnID ) ,
2010-07-11 03:34:59 +08:00
if EventID =:= false - > ignore ;
true - > send_1205 ( EventID , BlockID , 0 )
2010-06-19 00:33:13 +08:00
end ;
_ - >
ignore
end ;
2010-06-01 00:50:24 +08:00
2010-06-19 00:33:13 +08:00
%% @todo Handle this packet.
2010-07-11 03:34:59 +08:00
%% @todo 3rd Unsafe Passage C, EventID 10 BlockID 2 = mission cleared?
2010-06-28 07:33:31 +08:00
handle ( 16#0404 , Data ) - >
2010-07-11 03:34:59 +08:00
< < 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 ) ;
2010-06-01 00:50:24 +08:00
2010-08-17 00:31:01 +08:00
%% @todo Used in the tutorial. Not sure what it does. Give an item (the PA) maybe?
handle ( 16#0a09 , Data ) - >
log ( " ~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 > > ) ;
2010-07-31 04:09:09 +08:00
2010-05-24 19:30:36 +08:00
%% @doc Set flag handler. Associate a new flag with the character.
%% Just reply with a success value for now.
%% @todo God save the flags.
2010-06-28 07:33:31 +08:00
handle ( 16#0d04 , Data ) - >
2010-06-28 01:59:16 +08:00
< < Flag : 128 / bits , A : 16 / bits , _ : 8 , B / bits > > = Data ,
2010-06-28 07:33:31 +08:00
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 > > ) ;
2010-05-22 10:32:14 +08:00
2010-05-13 12:34:04 +08:00
%% @doc Options changes handler.
2010-07-01 00:02:00 +08:00
handle ( 16#0d07 , Data ) - >
2010-06-28 07:33:31 +08:00
log ( " options changes " ) ,
2010-07-01 00:02:00 +08:00
% Translate options into a tuple, validate them and do nothing with it for now
Options = psu_characters : options_binary_to_tuple ( Data ) ,
psu_characters : validate_options ( Options ) ,
% End of validation
2010-07-19 08:11:31 +08:00
{ 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 ] ) , Data ) ;
2010-05-13 12:34:04 +08:00
2010-07-27 01:37:50 +08:00
%% @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.
2010-08-23 05:00:43 +08:00
handle ( 16#1019 , _ ) - >
2010-08-23 04:34:44 +08:00
ignore ;
2010-07-27 01:37:50 +08:00
%~ send(<< (header(16#1019))/binary, 0:192, 16#00200000:32, 0:32 >>);
2010-07-17 20:50:03 +08:00
%% @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 ) ;
2010-07-27 01:37:50 +08:00
%% @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 ) ;
2010-06-03 02:49:19 +08:00
%% @doc Mission selected handler. Send the currently selected mission.
%% @todo Probably need to dispatch that info to other party members in the same counter.
2010-06-28 07:33:31 +08:00
handle ( 16#1707 , _ ) - >
2010-06-03 02:49:19 +08:00
ignore ;
2010-06-05 07:42:32 +08:00
%% @doc Counter-related handler.
2010-06-28 07:33:31 +08:00
handle ( 16#170b , _ ) - >
send_170c ( ) ;
2010-06-05 07:42:32 +08:00
2010-05-13 12:34:04 +08:00
%% @doc Unknown command handler. Do nothing.
2010-06-28 07:33:31 +08:00
handle ( Command , _ ) - >
log ( " dismissed packet ~4.16.0b " , [ Command ] ) .
2010-05-13 12:34:04 +08:00
2010-06-28 08:22:42 +08:00
%% @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 > > .
2010-06-28 07:33:31 +08:00
%% @doc Send the given packet to the client.
%% @todo Consolidate the receive and send functions better.
send ( Packet ) - >
2010-08-19 07:53:18 +08:00
psu_proto : packet_send ( get ( socket ) , Packet ) .
2010-06-02 02:11:22 +08:00
2010-07-02 08:43:10 +08:00
%% @todo Figure out what this does compared to 0201(self).
%% @todo Figure out the unknown values.
%% @todo Probably don't pattern match the data like this...
send_010d ( User ) - >
GID = get ( gid ) ,
2010-07-19 08:11:31 +08:00
CharGID = User #egs_user_model.id ,
2010-08-17 00:31:01 +08:00
CharLID = User #egs_user_model.lid ,
< < _ : 640 , CharBin / bits > > = psu_characters : character_user_to_binary ( User ) ,
2010-07-02 08:43:10 +08:00
send ( < < 16#010d0300 : 32 , 0 : 160 , 16#00011300 : 32 , GID : 32 / little - unsigned - integer , 0 : 64 ,
1 : 32 / little - unsigned - integer , 0 : 32 , 16#00000300 : 32 , 16#ffff0000 : 32 , 0 : 32 , CharGID : 32 / little - unsigned - integer ,
2010-08-17 00:31:01 +08:00
0 : 192 , CharGID : 32 / little - unsigned - integer , CharLID : 32 / little - unsigned - integer , 16#ffffffff : 32 , CharBin / binary > > ) .
2010-07-02 08:43:10 +08:00
2010-06-11 01:34:43 +08:00
%% @todo Possibly related to 010d. Just send seemingly safe values.
2010-07-11 04:18:28 +08:00
send_0111 ( A , B ) - >
2010-06-28 07:33:31 +08:00
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 ,
2010-07-11 04:18:28 +08:00
GID : 32 / little - unsigned - integer , 0 : 32 , A : 32 / little - unsigned - integer , B : 32 / little - unsigned - integer > > ) .
2010-06-11 01:34:43 +08:00
2010-06-14 05:42:56 +08:00
%% @todo Types capability list.
2010-06-28 07:33:31 +08:00
send_0113 ( ) - >
2010-06-14 05:42:56 +08:00
{ ok , File } = file : read_file ( " p/typesinfo.bin " ) ,
2010-06-28 07:33:31 +08:00
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 > > ) .
2010-06-14 05:42:56 +08:00
2010-07-02 04:47:02 +08:00
%% @doc Update the character's EXP.
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 > > ) .
2010-07-11 04:27:59 +08:00
%% @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 > > ) .
2010-06-11 01:34:43 +08:00
%% @doc Send the zone initialization notification.
2010-06-28 07:33:31 +08:00
send_0200 ( ZoneType ) - >
2010-06-11 01:34:43 +08:00
case ZoneType of
mission - >
Var = < < 16#06000500 : 32 , 16#01000000 : 32 , 0 : 64 , 16#00040000 : 32 , 16#00010000 : 32 , 16#00140000 : 32 > > ;
2010-06-13 03:26:37 +08:00
myroom - >
Var = < < 16#06000000 : 32 , 16#02000000 : 32 , 0 : 64 , 16#40000000 : 32 , 16#00010000 : 32 , 16#00010000 : 32 > > ;
2010-06-11 01:34:43 +08:00
_ - >
Var = < < 16#00040000 : 32 , 0 : 160 , 16#00140000 : 32 > >
end ,
2010-06-28 08:22:42 +08:00
send ( < < ( header ( 16#0200 ) ) / binary , 0 : 32 , 16#01000000 : 32 , 16#ffffffff : 32 , Var / binary , 16#ffffffff : 32 , 16#ffffffff : 32 > > ) .
2010-06-11 01:34:43 +08:00
2010-05-13 12:34:04 +08:00
%% @todo Figure out what the other things are.
2010-08-23 04:34:44 +08:00
%% @todo Handle LID correctly (should be ffffffff for self? yet LID should be 16 bits).
2010-07-02 03:26:26 +08:00
send_0201 ( User ) - >
2010-06-28 07:33:31 +08:00
GID = get ( gid ) ,
2010-07-19 08:11:31 +08:00
CharGID = User #egs_user_model.id ,
2010-08-17 00:31:01 +08:00
CharBin = psu_characters : character_user_to_binary ( User ) ,
2010-07-02 03:26:26 +08:00
IsGM = 0 ,
OnlineStatus = 0 ,
GameVersion = 0 ,
send ( < < 16#02010300 : 32 , 0 : 32 , 16#00001200 : 32 , CharGID : 32 / little - unsigned - integer , 0 : 64 , 16#00011300 : 32 ,
GID : 32 / little - unsigned - integer , 0 : 64 , CharBin / binary , IsGM : 8 , 0 : 8 , OnlineStatus : 8 , GameVersion : 8 , 0 : 608 > > ) .
2010-05-13 12:34:04 +08:00
2010-06-11 01:34:43 +08:00
%% @doc Hello packet, always sent on client connection.
2010-06-28 07:33:31 +08:00
send_0202 ( ) - >
send ( < < 16#02020300 : 32 , 0 : 352 > > ) .
2010-06-11 01:34:43 +08:00
2010-06-14 22:16:13 +08:00
%% @todo Not sure. Used for unspawning, and more.
2010-06-28 07:33:31 +08:00
send_0204 ( PlayerGID , PlayerLID , Action ) - >
GID = get ( gid ) ,
send ( < < 16#02040300 : 32 , 0 : 32 , 16#00001200 : 32 , PlayerGID : 32 / little - unsigned - integer , 0 : 64 ,
2010-06-14 22:16:13 +08:00
16#00011300 : 32 , GID : 32 / little - unsigned - integer , 0 : 64 , PlayerGID : 32 / little - unsigned - integer ,
2010-06-28 07:33:31 +08:00
PlayerLID : 32 / little - unsigned - integer , Action : 32 / little - unsigned - integer > > ) .
2010-06-14 22:16:13 +08:00
2010-06-11 01:34:43 +08:00
%% @doc Send the map ID to be loaded by the client.
%% @todo Last two values are unknown.
2010-06-28 07:33:31 +08:00
send_0205 ( MapType , MapNumber , MapEntry , IsSeasonal ) - >
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 > > ) .
2010-06-11 01:34:43 +08:00
%% @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.
2010-06-28 07:33:31 +08:00
send_0208 ( ) - >
2010-06-28 08:22:42 +08:00
send ( < < ( header ( 16#0208 ) ) / binary , 2 : 32 / little - unsigned - integer > > ) .
2010-06-11 01:34:43 +08:00
%% @todo No idea what this one does. For unknown reasons it uses channel 2.
2010-06-28 07:33:31 +08:00
send_020c ( ) - >
send ( < < 16#020c020c : 32 , 16#fffff20c : 32 , 0 : 256 > > ) .
2010-06-11 01:34:43 +08:00
%% @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.
2010-06-28 07:33:31 +08:00
send_020e ( Filename ) - >
2010-06-11 01:34:43 +08:00
{ ok , File } = file : read_file ( Filename ) ,
Size = byte_size ( File ) ,
2010-06-28 07:33:31 +08:00
send ( < < 16#020e0300 : 32 , 0 : 288 , Size : 32 / little - unsigned - integer , 0 : 32 , File / binary , 0 : 32 > > ) .
2010-06-11 01:34:43 +08:00
%% @doc Send the zone file to be loaded.
2010-07-10 00:54:37 +08:00
send_020f ( Filename , SetID , SeasonID ) - >
2010-06-11 01:34:43 +08:00
{ ok , File } = file : read_file ( Filename ) ,
Size = byte_size ( File ) ,
2010-07-10 00:54:37 +08:00
send ( < < 16#020f0300 : 32 , 0 : 288 , SetID , SeasonID , 0 : 16 , Size : 32 / little - unsigned - integer , File / binary > > ) .
2010-06-11 01:34:43 +08:00
2010-07-03 02:06:13 +08:00
%% @doc Send the current UNIX time.
send_0210 ( ) - >
CurrentTime = calendar : datetime_to_gregorian_seconds ( calendar : now_to_universal_time ( now ( ) ) )
- calendar : datetime_to_gregorian_seconds ( { { 1970 , 1 , 1 } , { 0 , 0 , 0 } } ) ,
send ( < < ( header ( 16#0210 ) ) / binary , 0 : 32 , CurrentTime : 32 / little - unsigned - integer > > ) .
2010-06-11 01:34:43 +08:00
%% @todo No idea what this do. Nor why it's sent twice when loading a counter.
2010-06-28 07:33:31 +08:00
send_0215 ( N ) - >
2010-06-28 08:22:42 +08:00
send ( < < ( header ( 16#0215 ) ) / binary , N : 32 / little - unsigned - integer > > ) .
2010-06-11 01:34:43 +08:00
%% @todo End of character loading. Just send it.
2010-06-28 07:33:31 +08:00
send_021b ( ) - >
2010-06-28 08:22:42 +08:00
send ( header ( 16#021b ) ) .
2010-06-11 01:34:43 +08:00
%% @doc Send the list of available universes.
2010-06-28 07:33:31 +08:00
send_021e ( ) - >
2010-07-19 08:11:31 +08:00
{ ok , Count } = egs_user_model : count ( ) ,
[ StrCount ] = io_lib : format ( " ~b " , [ Count ] ) ,
2010-08-01 00:04:39 +08:00
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 ] ) .
2010-06-11 01:34:43 +08:00
%% @doc Send the current universe name and number.
%% @todo Currently only have universe number 2, named EGS Test.
2010-06-28 07:33:31 +08:00
send_0222 ( ) - >
2010-06-28 04:07:03 +08:00
UCS2Name = < < < < X : 8 , 0 : 8 > > | | X < - " EGS Test " > > ,
2010-06-28 07:33:31 +08:00
GID = get ( gid ) ,
send ( < < 16#02220300 : 32 , 0 : 32 , 16#00001200 : 32 , GID : 32 / little - unsigned - integer , 0 : 64 , 16#00011300 : 32 , GID : 32 / little - unsigned - integer , 0 : 64 ,
2 : 32 / little - unsigned - integer , 0 : 32 , UCS2Name / binary , 0 : 16 > > ) .
2010-06-11 01:34:43 +08:00
2010-07-31 04:09:09 +08:00
%% @todo No idea!
send_022c ( A , B ) - >
send ( < < ( header ( 16#022c ) ) / binary , A : 16 / little - unsigned - integer , B : 16 / little - unsigned - integer > > ) .
2010-06-11 01:34:43 +08:00
%% @todo Not sure. Sent when going to or from room.
2010-06-28 07:33:31 +08:00
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 > > ) .
2010-06-11 01:34:43 +08:00
2010-05-15 05:45:03 +08:00
%% @todo Figure out what the other things are.
2010-06-28 07:33:31 +08:00
send_0233 ( Users ) - >
2010-05-15 05:45:03 +08:00
NbUsers = length ( Users ) ,
case NbUsers of
0 - >
ignore ;
_ - >
2010-06-28 07:33:31 +08:00
GID = get ( gid ) ,
2010-05-15 05:45:03 +08:00
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 > > ,
2010-06-11 01:34:43 +08:00
Contents = build_0233_contents ( Users ) ,
2010-06-28 07:33:31 +08:00
send ( < < Header / binary , Contents / binary > > )
2010-05-15 05:45:03 +08:00
end .
2010-07-02 03:26:26 +08:00
%% @todo God this function is ugly. Use tail recursion!
2010-06-11 01:34:43 +08:00
build_0233_contents ( [ ] ) - >
2010-05-15 05:45:03 +08:00
< < > > ;
2010-06-11 01:34:43 +08:00
build_0233_contents ( Users ) - >
2010-05-15 05:45:03 +08:00
[ User | Rest ] = Users ,
2010-08-23 04:34:44 +08:00
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...
2010-07-19 08:11:31 +08:00
CharBin = psu_characters : character_user_to_binary ( User #egs_user_model { lid = LID } ) ,
2010-07-02 03:26:26 +08:00
IsGM = 0 ,
GameVersion = 0 ,
Chunk = < < CharBin / binary , IsGM : 8 , 0 : 8 , GameVersion : 8 , 0 : 8 > > ,
2010-06-11 01:34:43 +08:00
Next = build_0233_contents ( Rest ) ,
2010-05-15 05:45:03 +08:00
< < Chunk / binary , Next / binary > > .
2010-06-11 01:34:43 +08:00
%% @doc Center the camera on the player, if possible.
%% @todo Probably.
2010-06-28 07:33:31 +08:00
send_0236 ( ) - >
2010-06-28 08:22:42 +08:00
send ( header ( 16#0236 ) ) .
2010-06-05 00:02:26 +08:00
2010-08-23 19:45:35 +08:00
%% @doc Send a chat command.
2010-08-17 00:31:01 +08:00
send_0304 ( FromTypeID , FromGID , FromName , Modifiers , Message ) - >
2010-08-23 20:45:47 +08:00
{ 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 > > ) .
2010-06-05 03:05:04 +08:00
2010-07-16 22:25:31 +08:00
%% @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 = _ } ) - >
2010-07-19 08:11:31 +08:00
{ 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 ,
2010-07-16 22:25:31 +08:00
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 > > ) .
2010-08-17 00:31:01 +08:00
%% @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 > > ) .
2010-06-05 00:30:57 +08:00
%% @todo Inventory related. No idea what it does.
2010-06-28 07:33:31 +08:00
send_0a05 ( ) - >
2010-06-28 08:22:42 +08:00
send ( header ( 16#0a05 ) ) .
2010-06-05 00:30:57 +08:00
2010-06-01 00:50:24 +08:00
%% @todo Inventory related. Figure out everything in this packet and handle it correctly.
2010-06-05 01:46:17 +08:00
%% @todo It sends 60 values so it's probably some kind of options for all 60 items in the inventory?
2010-06-28 07:33:31 +08:00
send_0a06 ( ) - >
2010-06-01 00:50:24 +08:00
{ ok , < < _ : 32 , A : 96 / bits , _ : 32 , B : 96 / bits , _ : 32 , C : 1440 / bits , _ : 32 , D / bits > > } = file : read_file ( " p/packet0a06.bin " ) ,
2010-06-28 07:33:31 +08:00
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 > > ) .
2010-06-01 00:50:24 +08:00
%% @todo Inventory. Figure out everything in this packet and handle it correctly.
2010-06-28 07:33:31 +08:00
send_0a0a ( ) - >
2010-06-01 00:50:24 +08:00
{ ok , < < _ : 32 , A : 224 / bits , _ : 32 , B / bits > > } = file : read_file ( " p/packet0a0a.bin " ) ,
2010-06-28 07:33:31 +08:00
GID = get ( gid ) ,
send ( < < A / binary , GID : 32 / little - unsigned - integer , B / binary > > ) .
2010-06-01 00:50:24 +08:00
2010-06-14 21:53:51 +08:00
%% @doc Item description.
2010-06-28 07:33:31 +08:00
send_0a11 ( ItemID , ItemDesc ) - >
2010-06-14 21:53:51 +08:00
Size = 1 + length ( ItemDesc ) ,
UCS2Desc = < < < < X : 8 , 0 : 8 > > | | X < - ItemDesc > > ,
2010-06-28 08:22:42 +08:00
send ( < < ( header ( 16#0a11 ) ) / binary , ItemID : 32 / unsigned - integer , Size : 32 / little - unsigned - integer , UCS2Desc / binary , 0 : 16 > > ) .
2010-06-14 21:53:51 +08:00
2010-06-11 01:34:43 +08:00
%% @doc Init quest.
2010-06-28 07:33:31 +08:00
send_0c00 ( QuestID ) - >
2010-06-28 08:22:42 +08:00
send ( < < ( header ( 16#0c00 ) ) / binary , QuestID : 32 / little - unsigned - integer ,
2010-06-11 01:34:43 +08:00
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 ,
16#ffffffff : 32 , 16#ffffffff : 32 , 16#ffffffff : 32 , 16#ffffffff : 32 , 16#ffffffff : 32 , 16#ffffffff : 32 , 16#ffffffff : 32 , 16#ffffffff : 32 ,
2010-06-28 07:33:31 +08:00
16#ffffffff : 32 , 16#ffffffff : 32 , 16#ffffffff : 32 , 16#ffffffff : 32 , 16#ffffffff : 32 , 16#ffffffff : 32 , 16#ffffffff : 32 , 16#ffffffff : 32 > > ) .
2010-06-11 01:34:43 +08:00
2010-06-11 22:55:32 +08:00
%% @todo Figure out last 4 bytes!
2010-06-28 07:33:31 +08:00
send_0c02 ( ) - >
2010-06-28 08:22:42 +08:00
send ( < < ( header ( 16#0c02 ) ) / binary , 0 : 32 > > ) .
2010-06-11 22:55:32 +08:00
2010-06-11 23:42:58 +08:00
%% @doc Send the huge pack of quest files available in the counter.
2010-06-28 07:33:31 +08:00
send_0c06 ( Filename ) - >
2010-06-11 23:42:58 +08:00
{ ok , < < File / bits > > } = file : read_file ( Filename ) ,
2010-06-28 07:33:31 +08:00
send ( < < 16#0c060300 : 32 , 0 : 288 , 1 : 32 / little - unsigned - integer , File / binary > > ) .
2010-06-11 23:42:58 +08:00
2010-06-11 23:16:52 +08:00
%% @doc Reply whether the player is allowed to use the transport option.
2010-06-28 18:36:56 +08:00
%% Use true for allowing it, and false otherwise.
2010-06-28 07:33:31 +08:00
send_0c08 ( Response ) - >
2010-06-28 18:36:56 +08:00
Value = if Response =:= true - > 0 ; true - > 1 end ,
2010-06-28 08:22:42 +08:00
send ( < < ( header ( 16#0c08 ) ) / binary , Value : 32 > > ) .
2010-06-11 23:16:52 +08:00
2010-06-11 01:34:43 +08:00
%% @doc Send the trial start notification.
2010-06-28 07:33:31 +08:00
send_0c09 ( ) - >
2010-06-28 08:22:42 +08:00
send ( < < ( header ( 16#0c09 ) ) / binary , 0 : 64 > > ) .
2010-06-11 01:34:43 +08:00
2010-06-11 23:42:58 +08:00
%% @doc Send the counter's mission options (0 = invisible, 2 = disabled, 3 = available).
2010-06-28 07:33:31 +08:00
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 > > ) .
2010-06-11 23:42:58 +08:00
2010-06-11 01:34:43 +08:00
%% @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.
2010-07-02 03:26:26 +08:00
send_0d01 ( User ) - >
2010-07-19 08:11:31 +08:00
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 ) ,
2010-07-02 03:26:26 +08:00
send ( < < ( header ( 16#0d01 ) ) / binary , CharBin / binary ,
2010-06-11 01:34:43 +08:00
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 ,
2010-07-02 03:26:26 +08:00
0 : 7744 , OptionsBin / binary > > ) .
2010-06-11 01:34:43 +08:00
%% @doc Send the character list for selection.
2010-07-03 01:15:45 +08:00
%% @todo There's a few odd values blanked, also the last known location apparently.
2010-06-28 07:33:31 +08:00
send_0d03 ( Data0 , Data1 , Data2 , Data3 ) - >
2010-06-11 01:34:43 +08:00
[ { status , Status0 } , { char , Char0 } | _ ] = Data0 ,
[ { status , Status1 } , { char , Char1 } | _ ] = Data1 ,
[ { status , Status2 } , { char , Char2 } | _ ] = Data2 ,
[ { status , Status3 } , { char , Char3 } | _ ] = Data3 ,
2010-06-28 07:33:31 +08:00
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 ,
2010-06-11 01:34:43 +08:00
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 ,
2010-06-28 07:33:31 +08:00
Status3 : 8 / unsigned - integer , 0 : 48 , Char3 / binary , 0 : 512 > > ) .
2010-06-11 01:34:43 +08:00
%% @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.
2010-06-28 07:33:31 +08:00
send_0d05 ( ) - >
2010-06-11 01:34:43 +08:00
{ ok , Flags } = file : read_file ( " p/flags.bin " ) ,
2010-06-28 07:33:31 +08:00
GID = get ( gid ) ,
2010-06-11 01:34:43 +08:00
Packet = < < 16#0d050300 : 32 , 0 : 32 , 16#00011300 : 32 , GID : 32 / little - unsigned - integer , 0 : 64 , 16#00011300 : 32 , GID : 32 / little - unsigned - integer , 0 : 64 , Flags / binary > > ,
Size = 4 + byte_size ( Packet ) ,
2010-06-28 07:33:31 +08:00
ssl : send ( get ( socket ) , < < Size : 32 / little - unsigned - integer , Packet / binary > > ) .
2010-06-11 01:34:43 +08:00
2010-07-31 04:09:09 +08:00
%% @todo Add a character (NPC or real) to the party members on the right of the screen.
%% @todo NPCid is 65535 for normal characters.
2010-08-17 00:31:01 +08:00
%% @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 ,
#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 ( < < ( header ( 16#1004 ) ) / binary , TypeID : 32 ,
GID : 32 / little - unsigned - integer , 0 : 64 , Name / binary ,
2010-07-31 21:20:25 +08:00
Level : 16 / little - unsigned - integer , 16#ffff : 16 ,
2010-08-17 00:31:01 +08:00
SomeFlag , 1 , PartyPos : 8 , 1 ,
2010-07-31 21:20:25 +08:00
NPCid : 16 / little - unsigned - integer , 0 : 16 ,
2010-08-17 00:31:01 +08:00
%% Odd unknown values. PA related? No idea. Values on invite, 0 in-mission.
2010-07-31 21:20:25 +08:00
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 ,
2010-08-17 00:31:01 +08:00
16#64e41f08 : 32 , 0 : 32 , 16#12000000 : 32 ,
16#24e41f08 : 32 ,
2010-07-31 21:20:25 +08:00
QuestID : 32 / little - unsigned - integer , ZoneID : 32 / little - unsigned - integer , MapID : 32 / little - unsigned - integer , EntryID : 32 / little - unsigned - integer ,
2010-08-17 00:31:01 +08:00
LID : 32 ,
0 : 64 ,
16#01000000 : 32 , 16#01000000 : 32 , %% @todo first is current hp, second is max hp
2010-07-31 21:20:25 +08:00
0 : 608 > > ) .
2010-07-31 04:09:09 +08:00
2010-05-13 12:34:04 +08:00
%% @todo Figure out what the packet is.
2010-07-02 03:26:26 +08:00
send_1005 ( Name ) - >
2010-05-13 12:34:04 +08:00
{ ok , File } = file : read_file ( " p/packet1005.bin " ) ,
< < _ : 352 , Before : 160 / bits , _ : 608 , After / bits > > = File ,
2010-06-28 07:33:31 +08:00
GID = get ( gid ) ,
2010-06-28 08:22:42 +08:00
send ( < < ( header ( 16#1005 ) ) / binary , Before / binary , GID : 32 / little - unsigned - integer , 0 : 64 , Name / binary , After / binary > > ) .
2010-05-13 12:34:04 +08:00
2010-06-11 04:50:32 +08:00
%% @doc Party-related command probably controlling the party state.
%% Value 11 aborts the mission.
2010-06-04 22:25:38 +08:00
%% @todo Figure out what the packet is.
2010-06-28 07:33:31 +08:00
send_1006 ( N ) - >
2010-06-28 08:22:42 +08:00
send ( < < ( header ( 16#1006 ) ) / binary , N : 32 / little - unsigned - integer > > ) .
2010-06-04 22:25:38 +08:00
2010-06-11 01:34:43 +08:00
%% @doc Send the player's current location.
2010-06-28 07:33:31 +08:00
send_100e ( QuestID , ZoneID , MapID , Location , CounterID ) - >
2010-06-11 01:34:43 +08:00
UCS2Location = < < < < X : 8 , 0 : 8 > > | | X < - Location > > ,
2010-06-28 08:22:42 +08:00
Packet = < < ( header ( 16#100e ) ) / binary , 1 : 32 / little - unsigned - integer , MapID : 16 / little - unsigned - integer ,
ZoneID : 16 / little - unsigned - integer , QuestID : 32 / little - unsigned - integer , UCS2Location / binary > > ,
2010-06-11 01:34:43 +08:00
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 ,
2010-06-28 07:33:31 +08:00
send ( < < Packet / binary , 0 : PaddingSize , Footer / binary > > ) .
2010-06-11 01:34:43 +08:00
2010-08-23 04:34:44 +08:00
%% @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.
2010-07-31 21:20:25 +08:00
send_100f ( NPCid , PartyPos ) - >
send ( < < ( header ( 16#100f ) ) / binary , NPCid : 16 / little - unsigned - integer , 1 , PartyPos : 8 , PartyPos : 32 / little - unsigned - integer > > ) .
2010-06-09 01:09:54 +08:00
%% @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.
2010-06-28 07:33:31 +08:00
send_1015 ( QuestID ) - >
2010-06-10 00:40:19 +08:00
[ { type , _ } , { file , QuestFile } | _ ] = proplists : get_value ( QuestID , ? QUESTS ) ,
2010-06-09 01:09:54 +08:00
{ ok , File } = file : read_file ( QuestFile ) ,
Size = byte_size ( File ) ,
2010-06-28 08:22:42 +08:00
send ( < < ( header ( 16#1015 ) ) / binary , QuestID : 32 / little - unsigned - integer , 16#01010000 : 32 , 0 : 32 , Size : 32 / little - unsigned - integer , File / binary > > ) .
2010-06-05 18:46:04 +08:00
2010-07-31 21:20:25 +08:00
%% @todo No idea.
send_1016 ( PartyPos ) - >
send ( < < ( header ( 16#1016 ) ) / binary , PartyPos : 32 / little - unsigned - integer > > ) .
2010-08-17 00:31:01 +08:00
%% @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 > > ) .
2010-06-11 22:55:32 +08:00
%% @todo Totally unknown.
2010-06-28 07:33:31 +08:00
send_1020 ( ) - >
2010-06-28 08:22:42 +08:00
send ( header ( 16#1020 ) ) .
2010-06-11 22:55:32 +08:00
2010-07-11 04:27:59 +08:00
%% @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 > > ) .
2010-07-17 20:50:03 +08:00
%% @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 > > ) .
2010-06-05 07:15:50 +08:00
%% @todo Figure out what this packet does. Sane values for counter and missions for now.
2010-06-28 07:33:31 +08:00
send_1202 ( ) - >
2010-06-28 08:22:42 +08:00
send ( < < ( header ( 16#1202 ) ) / binary , 0 : 32 , 16#10000000 : 32 , 0 : 64 , 16#14000000 : 32 , 0 : 32 > > ) .
2010-06-05 07:15:50 +08:00
%% @todo Figure out what this packet does. Seems it's the same values all the time.
2010-06-28 07:33:31 +08:00
send_1204 ( ) - >
2010-06-28 08:22:42 +08:00
send ( < < ( header ( 16#1204 ) ) / binary , 0 : 32 , 16#20000000 : 32 , 0 : 256 > > ) .
2010-06-05 07:15:50 +08:00
2010-06-19 00:33:13 +08:00
%% @doc Object events response?
2010-07-11 03:34:59 +08:00
%% @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 > > ) .
2010-06-19 00:33:13 +08:00
2010-06-05 07:15:50 +08:00
%% @todo Figure out what this packet does. Sane values for counter and missions for now.
2010-06-28 07:33:31 +08:00
send_1206 ( ) - >
2010-06-28 08:22:42 +08:00
send ( < < ( header ( 16#1206 ) ) / binary , 0 : 32 , 16#80020000 : 32 , 0 : 5120 > > ) .
2010-06-05 07:15:50 +08:00
%% @todo Figure out what this packet does. Sane values for counter and missions for now.
2010-06-28 07:33:31 +08:00
send_1207 ( ) - >
2010-06-05 07:15:50 +08:00
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 > > ,
2010-06-28 08:22:42 +08:00
send ( < < ( header ( 16#1207 ) ) / binary , Chunk / binary , Chunk / binary , Chunk / binary , Chunk / binary , Chunk / binary , Chunk / binary > > ) .
2010-06-05 07:15:50 +08:00
2010-06-14 21:38:57 +08:00
%% @todo Object interaction? Figure out. C probably the interaction type.
2010-08-25 07:42:37 +08:00
%% @todo Apparently A would be TargetID/ffffffff, B would be the player LID, C would be the object type? D still completely unknown.
2010-06-28 07:33:31 +08:00
send_1211 ( A , B , C , D ) - >
2010-06-28 08:22:42 +08:00
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 > > ) .
2010-06-14 21:38:57 +08:00
2010-06-11 01:34:43 +08:00
%% @doc Make the client load the quest previously sent.
2010-06-28 07:33:31 +08:00
send_1212 ( ) - >
2010-06-28 08:22:42 +08:00
send ( < < ( header ( 16#1212 ) ) / binary , 0 : 19200 > > ) .
2010-06-11 01:34:43 +08:00
2010-06-19 00:33:13 +08:00
%% @todo Not sure. Related to keys.
2010-06-28 07:33:31 +08:00
send_1213 ( A , B ) - >
2010-06-28 08:22:42 +08:00
send ( < < ( header ( 16#1213 ) ) / binary , A : 32 / little - unsigned - integer , B : 32 / little - unsigned - integer > > ) .
2010-06-19 00:33:13 +08:00
2010-07-17 20:50:03 +08:00
%% @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 > > ) .
2010-07-27 01:37:50 +08:00
%% @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 > > ) .
2010-08-01 00:28:31 +08:00
%% @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 ) .
2010-06-11 01:34:43 +08:00
%% @doc Send the player's partner card.
2010-07-02 03:26:26 +08:00
%% @todo Find out the remaining values.
send_1500 ( User ) - >
2010-07-19 08:11:31 +08:00
#characters { slot = Slot , name = Name , race = Race , gender = Gender , class = Class } = User #egs_user_model.character ,
2010-07-02 03:26:26 +08:00
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 ( < < ( header ( 16#1500 ) ) / binary , Name / binary , RaceBin : 8 , GenderBin : 8 , ClassBin : 8 , 0 : 3112 , 16#010401 : 24 , Slot : 8 , 0 : 64 > > ) .
2010-06-11 01:34:43 +08:00
2010-06-04 22:53:32 +08:00
%% @todo Send an empty partner card list.
2010-06-28 07:33:31 +08:00
send_1501 ( ) - >
2010-06-28 08:22:42 +08:00
send ( < < ( header ( 16#1501 ) ) / binary , 0 : 32 > > ) .
2010-06-04 22:53:32 +08:00
2010-06-04 23:02:13 +08:00
%% @todo Send an empty blacklist.
2010-06-28 07:33:31 +08:00
send_1512 ( ) - >
2010-06-28 08:22:42 +08:00
send ( < < ( header ( 16#1512 ) ) / binary , 0 : 46080 > > ) .
2010-06-04 23:02:13 +08:00
2010-08-17 00:31:01 +08:00
%% @todo NPC related packet, sent when there's an NPC in the area.
send_1601 ( ) - >
{ ok , Bin } = file : read_file ( " p/packet1601.bin " ) ,
send ( < < ( header ( 16#1601 ) ) / binary , Bin / binary > > ) .
2010-06-11 01:34:43 +08:00
%% @doc Send the player's NPC and PM information.
2010-07-31 01:31:31 +08:00
%% @todo The value 4 is the card priority. Find what 3 is. When sending, the first 0 is an unknown value.
2010-06-28 07:33:31 +08:00
send_1602 ( ) - >
2010-07-31 01:31:31 +08:00
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 ) ) ,
send ( < < ( header ( 16#1602 ) ) / binary , 0 : 32 , Bin / binary , 0 : MiddlePaddingSize , NbNPC , 0 : 24 , UCS2PMName / binary , 0 : EndPaddingSize > > ) .
2010-06-11 01:34:43 +08:00
2010-06-11 22:41:23 +08:00
%% @doc Party information.
%% @todo Handle existing parties.
2010-06-28 07:33:31 +08:00
send_1706 ( CharName ) - >
2010-06-28 08:22:42 +08:00
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 > > ) .
2010-06-11 22:41:23 +08:00
%% @doc Party settings. Item distribution is random for now.
%% @todo Handle correctly.
2010-06-28 07:33:31 +08:00
send_170a ( ) - >
2010-06-28 08:22:42 +08:00
send ( < < ( header ( 16#170a ) ) / binary , 16#01010c08 : 32 > > ) .
2010-06-11 22:41:23 +08:00
2010-06-05 18:46:04 +08:00
%% @todo Find what the heck this packet is.
2010-06-28 07:33:31 +08:00
send_170c ( ) - >
2010-06-05 18:46:04 +08:00
{ ok , File } = file : read_file ( " p/packet170c.bin " ) ,
2010-06-28 08:22:42 +08:00
send ( < < ( header ( 16#170c ) ) / binary , File / binary > > ) .
2010-06-11 22:34:46 +08:00
2010-06-11 22:41:23 +08:00
%% @doc Send the background to use for the counter.
%% @todo Background has more info past the first byte.
2010-06-28 07:33:31 +08:00
send_1711 ( Background ) - >
2010-06-28 08:22:42 +08:00
send ( < < ( header ( 16#1711 ) ) / binary , Background : 32 / little - unsigned - integer > > ) .
2010-06-11 22:41:23 +08:00
2010-06-11 22:34:46 +08:00
%% @doc Unknown dialog-related handler.
%% @todo Everything!
2010-06-28 07:33:31 +08:00
send_1a02 ( A , B , C , D , E ) - >
2010-06-28 08:22:42 +08:00
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 > > ) .
2010-06-11 22:34:46 +08:00
%% @doc Lumilass handler. Possibly more.
%% @todo Figure out how Lumilass work exactly. The 4 bytes before the file may vary.
2010-06-28 07:33:31 +08:00
send_1a03 ( ) - >
2010-06-11 22:34:46 +08:00
{ ok , File } = file : read_file ( " p/lumilassA.bin " ) ,
2010-06-28 08:22:42 +08:00
send ( < < ( header ( 16#1a03 ) ) / binary , 0 : 32 , File / binary > > ) .
2010-06-11 22:34:46 +08:00
2010-06-28 08:22:42 +08:00
%% @doc PP cube handler.
2010-06-11 22:34:46 +08:00
%% @todo The 4 bytes before the file may vary. Everything past that is the same. Figure things out.
2010-06-28 07:33:31 +08:00
send_1a04 ( ) - >
2010-06-11 22:34:46 +08:00
{ ok , File } = file : read_file ( " p/ppcube.bin " ) ,
2010-06-28 08:22:42 +08:00
send ( < < ( header ( 16#1a04 ) ) / binary , 0 : 32 , File / binary > > ) .
2010-06-14 05:42:56 +08:00
%% @doc Types menu handler.
%% @todo Handle correctly.
2010-06-28 07:33:31 +08:00
send_1a07 ( ) - >
2010-06-28 08:22:42 +08:00
send ( < < ( header ( 16#1a07 ) ) / binary , 16#085b5d0a : 32 , 16#3a200000 : 32 , 0 : 32 ,
2010-06-28 07:33:31 +08:00
16#01010101 : 32 , 16#01010101 : 32 , 16#01010101 : 32 , 16#01010101 : 32 > > ) .
2010-06-05 18:46:04 +08:00
2010-05-13 12:34:04 +08:00
%% @doc Log message to the console.
2010-06-28 07:33:31 +08:00
log ( Message ) - >
io : format ( " game ( ~p ): ~s ~n " , [ get ( gid ) , Message ] ) .
2010-06-04 05:36:34 +08:00
2010-06-28 07:33:31 +08:00
log ( Message , Format ) - >
FormattedMessage = io_lib : format ( Message , Format ) ,
log ( FormattedMessage ) .