2010-09-19 01:57:55 +08:00
%% @author Lo<4C> c Hoguin <essen@dev-extend.eu>
%% @copyright 2010 Lo<4C> c Hoguin.
2010-09-19 04:53:15 +08:00
%% @doc Game callback module.
2010-09-19 01:57:55 +08:00
%%
%% This file is part of EGS.
%%
%% EGS is free software: you can redistribute it and/or modify
%% it under the terms of the GNU Affero General Public License as
%% published by the Free Software Foundation, either version 3 of the
%% License, or (at your option) any later version.
%%
%% EGS is distributed in the hope that it will be useful,
%% but WITHOUT ANY WARRANTY; without even the implied warranty of
%% MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
%% GNU Affero General Public License for more details.
%%
%% You should have received a copy of the GNU Affero General Public License
%% along with EGS. If not, see <http://www.gnu.org/licenses/>.
- module ( egs_game ) .
- export ( [ keepalive / 1 , info / 2 , cast / 3 , raw / 3 , event / 2 ] ) .
- include ( " include/records.hrl " ) .
%% @doc Send a keepalive.
keepalive ( #state { socket = Socket } ) - >
psu_proto : send_keepalive ( Socket ) .
%% @doc Forward the broadcasted command to the client.
info ( { egs , cast , Command } , #state { gid = GID } ) - >
< < A : 64 / bits , _ : 32 , B : 96 / bits , _ : 64 , C / bits > > = Command ,
psu_game : send ( < < A / binary , 16#00011300 : 32 , B / binary , 16#00011300 : 32 , GID : 32 / little - unsigned - integer , C / binary > > ) ;
%% @doc Forward the chat message to the client.
2010-10-20 00:12:25 +08:00
info ( { egs , chat , FromGID , ChatTypeID , ChatGID , ChatName , ChatModifiers , ChatMessage } , State ) - >
psu_proto : send_0304 ( FromGID , ChatTypeID , ChatGID , ChatName , ChatModifiers , ChatMessage , State ) ;
2010-09-19 01:57:55 +08:00
2010-10-09 08:07:02 +08:00
info ( { egs , notice , Type , Message } , State ) - >
psu_proto : send_0228 ( Type , 2 , Message , State ) ;
2010-09-19 01:57:55 +08:00
%% @doc Inform the client that a player has spawn.
2010-10-22 10:04:21 +08:00
%% @todo Not sure what IsSeasonal or the AreaNb in 0205 should be for other spawns.
info ( { egs , player_spawn , Player } , State ) - >
psu_proto : send_0111 ( Player , 6 , State ) ,
psu_proto : send_010d ( Player , State ) ,
psu_proto : send_0205 ( Player , 0 , State ) ,
psu_proto : send_0203 ( Player , State ) ,
psu_proto : send_0201 ( Player , State ) ;
2010-09-19 01:57:55 +08:00
%% @doc Inform the client that a player has unspawn.
2010-10-22 04:31:06 +08:00
info ( { egs , player_unspawn , Player } , State ) - >
psu_proto : send_0204 ( Player , State ) ;
2010-09-19 01:57:55 +08:00
%% @doc Warp the player to the given location.
info ( { egs , warp , QuestID , ZoneID , MapID , EntryID } , State ) - >
event ( { area_change , QuestID , ZoneID , MapID , EntryID } , State ) .
%% Broadcasts.
%% @todo Handle broadcasting better than that. Review the commands at the same time.
%% @doc Position change. Save the position and then dispatch it.
cast ( 16#0503 , Data , State = #state { gid = GID } ) - >
< < _ : 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 > > = Data ,
FloatDir = Dir / 46603 . 375 ,
2011-02-13 00:53:58 +08:00
{ ok , User } = egs_users : read ( GID ) ,
2011-02-13 06:28:56 +08:00
NewUser = User #users { pos = { X , Y , Z , FloatDir } , area = { QuestID , ZoneID , MapID } , entryid = EntryID } ,
2011-02-13 00:53:58 +08:00
egs_users : write ( NewUser ) ,
2010-09-19 01:57:55 +08:00
cast ( valid , Data , State ) ;
%% @doc Stand still. Save the position and then dispatch it.
cast ( 16#0514 , Data , State = #state { gid = GID } ) - >
< < _ : 424 , Dir : 24 / little - unsigned - integer , X : 32 / little - float , Y : 32 / little - float , Z : 32 / little - float ,
QuestID : 32 / little - unsigned - integer , ZoneID : 32 / little - unsigned - integer ,
MapID : 32 / little - unsigned - integer , EntryID : 32 / little - unsigned - integer , _ / bits > > = Data ,
FloatDir = Dir / 46603 . 375 ,
2011-02-13 00:53:58 +08:00
{ ok , User } = egs_users : read ( GID ) ,
2011-02-13 06:28:56 +08:00
NewUser = User #users { pos = { X , Y , Z , FloatDir } , area = { QuestID , ZoneID , MapID } , entryid = EntryID } ,
2011-02-13 00:53:58 +08:00
egs_users : write ( NewUser ) ,
2010-09-19 01:57:55 +08:00
cast ( valid , Data , State ) ;
%% @doc Default broadcast handler. Dispatch the command to everyone.
%% We clean up the command and use the real GID and LID of the user, disregarding what was sent and possibly tampered with.
%% Only a handful of commands are allowed to broadcast. An user tampering with it would get disconnected instantly.
%% @todo Don't query the user data everytime! Keep the needed information in the State.
cast ( Command , Data , #state { gid = GID } )
when Command =:= 16#0101 ;
Command =:= 16#0102 ;
Command =:= 16#0104 ;
Command =:= 16#0107 ;
Command =:= 16#010f ;
Command =:= 16#050f ;
Command =:= valid - >
< < _ : 32 , A : 64 / bits , _ : 64 , B : 192 / bits , _ : 64 , C / bits > > = Data ,
2011-02-13 00:53:58 +08:00
case egs_users : read ( GID ) of
2010-09-19 01:57:55 +08:00
{ error , _ Reason } - >
ignore ;
{ ok , Self } - >
2011-02-13 00:53:58 +08:00
LID = Self #users.lid ,
2010-09-19 01:57:55 +08:00
Packet = < < A / binary , 16#00011300 : 32 , GID : 32 / little - unsigned - integer , B / binary ,
GID : 32 / little - unsigned - integer , LID : 32 / little - unsigned - integer , C / binary > > ,
2011-02-13 00:53:58 +08:00
{ ok , SpawnList } = egs_users : select ( { neighbors , Self } ) ,
lists : foreach ( fun ( User ) - > User #users.pid ! { egs , cast , Packet } end , SpawnList )
2010-09-19 01:57:55 +08:00
end .
%% Raw commands.
%% @todo Handle this packet properly.
%% @todo Spawn cleared response event shouldn't be handled following this packet but when we see the spawn actually dead HP-wise.
%% @todo Type shouldn't be :32 but it seems when the later 16 have something it's not a spawn event.
raw ( 16#0402 , < < _ : 352 , Data / bits > > , #state { gid = GID } ) - >
< < SpawnID : 32 / little - unsigned - integer , _ : 64 , Type : 32 / little - unsigned - integer , _ : 64 > > = Data ,
case Type of
7 - > % spawn cleared @todo 1201 sent back with same values apparently, but not always
log ( " cleared spawn ~b " , [ SpawnID ] ) ,
2011-02-13 00:53:58 +08:00
{ ok , User } = egs_users : read ( GID ) ,
2011-02-13 06:28:56 +08:00
{ BlockID , EventID } = psu_instance : spawn_cleared_event ( User #users.instancepid , element ( 2 , User #users.area ) , SpawnID ) ,
2010-09-19 01:57:55 +08:00
if EventID =:= false - > ignore ;
true - > psu_game : send_1205 ( EventID , BlockID , 0 )
end ;
_ - >
ignore
end ;
%% @todo Handle this packet.
%% @todo 3rd Unsafe Passage C, EventID 10 BlockID 2 = mission cleared?
raw ( 16#0404 , < < _ : 352 , Data / bits > > , _ State ) - >
< < EventID : 8 , BlockID : 8 , _ : 16 , Value : 8 , _ / bits > > = Data ,
log ( " unknown command 0404: eventid ~b blockid ~b value ~b " , [ EventID , BlockID , Value ] ) ,
psu_game : send_1205 ( EventID , BlockID , Value ) ;
%% @todo Used in the tutorial. Not sure what it does. Give an item (the PA) maybe?
2010-09-29 05:06:27 +08:00
%% @todo Probably should ignore that until more is known.
2010-09-19 01:57:55 +08:00
raw ( 16#0a09 , _ Data , #state { gid = GID } ) - >
psu_game : send ( < < 16#0a090300 : 32 , 0 : 32 , 16#00011300 : 32 , GID : 32 / little - unsigned - integer , 0 : 64 , 16#00011300 : 32 , GID : 32 / little - unsigned - integer , 0 : 64 , 16#00003300 : 32 , 0 : 32 > > ) ;
%% @todo Figure out this command.
raw ( 16#0c11 , < < _ : 352 , A : 32 / little , B : 32 / little > > , #state { gid = GID } ) - >
log ( " 0c11 ~p ~p " , [ A , B ] ) ,
psu_game : send ( < < 16#0c120300 : 32 , 0 : 160 , 16#00011300 : 32 , GID : 32 / little - unsigned - integer , 0 : 64 , A : 32 / little , 1 : 32 / little > > ) ;
%% @doc Set flag handler. Associate a new flag with the character.
%% Just reply with a success value for now.
%% @todo God save the flags.
raw ( 16#0d04 , < < _ : 352 , Data / bits > > , #state { gid = GID } ) - >
< < Flag : 128 / bits , A : 16 / bits , _ : 8 , B / bits > > = Data ,
log ( " flag handler for ~s " , [ re : replace ( Flag , " \\ 0+ " , " " , [ global , { return , binary } ] ) ] ) ,
psu_game : send ( < < 16#0d040300 : 32 , 0 : 160 , 16#00011300 : 32 , GID : 32 / little - unsigned - integer , 0 : 64 , Flag / binary , A / binary , 1 , B / binary > > ) ;
%% @doc Initialize a vehicle object.
%% @todo Find what are the many values, including the odd Whut value (and whether it's used in the reply).
%% @todo Separate the reply.
raw ( 16#0f00 , < < _ : 352 , Data / bits > > , _ State ) - >
< < 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 ] ) ,
psu_game : send ( < < ( psu_game : 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.
raw ( 16#0f02 , < < _ : 352 , Data / bits > > , _ State ) - >
< < 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 ,
psu_game : send ( < < ( psu_game : 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.
raw ( 16#0f07 , < < _ : 352 , Data / bits > > , _ State ) - >
< < A : 32 / little - unsigned - integer , B : 32 / little - unsigned - integer > > = Data ,
log ( " after enter vehicle: ~b ~b " , [ A , B ] ) ,
psu_game : send ( < < ( psu_game : header ( 16#120f ) ) / binary , A : 32 / little - unsigned - integer , B : 32 / little - unsigned - integer > > ) ;
%% @todo Not sure yet.
raw ( 16#1019 , _ Data , _ State ) - >
ignore ;
%~ psu_game:send(<< (psu_game:header(16#1019))/binary, 0:192, 16#00200000:32, 0:32 >>);
%% @todo Not sure about that one though. Probably related to 1112 still.
raw ( 16#1106 , < < _ : 352 , Data / bits > > , _ State ) - >
psu_game : send_110e ( Data ) ;
%% @doc Probably asking permission to start the video (used for syncing?).
raw ( 16#1112 , < < _ : 352 , Data / bits > > , _ State ) - >
psu_game : send_1113 ( Data ) ;
%% @todo Not sure yet. Value is probably a TargetID. Used in Airboard Rally. Replying with the same value starts the race.
raw ( 16#1216 , < < _ : 352 , Data / bits > > , _ State ) - >
< < Value : 32 / little - unsigned - integer > > = Data ,
log ( " command 1216 with value ~b " , [ Value ] ) ,
psu_game : send_1216 ( Value ) ;
%% @doc Dismiss all unknown raw commands with a log notice.
%% @todo Have a log event handler instead.
raw ( Command , _ Data , State ) - >
io : format ( " ~p ( ~p ): dismissed command ~4.16.0b ~n " , [ ? MODULE , State #state.gid , Command ] ) .
%% Events.
2010-10-21 23:00:30 +08:00
%% @todo When changing lobby to the room, or room to lobby, we must perform an universe change.
2010-09-19 01:57:55 +08:00
%% @todo Probably move area_load inside the event and make other events call this one when needed.
event ( { area_change , QuestID , ZoneID , MapID , EntryID } , State ) - >
event ( { area_change , QuestID , ZoneID , MapID , EntryID , 16#ffffffff } , State ) ;
2010-09-25 22:51:07 +08:00
event ( { area_change , QuestID , ZoneID , MapID , EntryID , PartyPos } , State ) - >
2010-09-19 01:57:55 +08:00
case PartyPos of
16#ffffffff - >
log ( " area change ( ~b , ~b , ~b , ~b , ~b ) " , [ QuestID , ZoneID , MapID , EntryID , PartyPos ] ) ,
2010-09-25 22:51:07 +08:00
psu_game : area_load ( QuestID , ZoneID , MapID , EntryID , State ) ;
2010-09-19 01:57:55 +08:00
_ Any - > %% @todo Handle area_change event for NPCs in story missions.
ignore
end ;
%% @doc After the character has been (re)loaded, change the area he's in.
%% @todo The area_load function should probably not change the user's values.
%% @todo Remove that ugly code when the above is done.
event ( char_load_complete , State = #state { gid = GID } ) - >
2011-02-13 06:28:56 +08:00
{ ok , User = #users { area = { QuestID , ZoneID , MapID } , entryid = EntryID } } = egs_users : read ( GID ) ,
egs_users : write ( User #users { area = { 0 , 0 , 0 } , entryid = 0 } ) ,
2010-09-19 01:57:55 +08:00
event ( { area_change , QuestID , ZoneID , MapID , EntryID } , State ) ;
%% @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 } , #state { gid = UserGID } ) - >
[ BcastTypeID , BcastGID , BcastName ] = case FromGID of
0 - > %% This probably shouldn't happen. Just make it crash on purpose.
log ( " chat FromGID=0 " ) ,
ignore ;
UserGID - > %% player chat: disregard whatever was sent except modifiers and message.
2011-02-13 00:53:58 +08:00
{ ok , User } = egs_users : read ( UserGID ) ,
2011-02-13 07:35:05 +08:00
[ 16#00001200 , User #users.gid , ( User #users.character ) #characters.name ] ;
2010-09-19 01:57:55 +08:00
NPCGID - > %% npc chat: @todo Check that the player is the party leader and this npc is in his party.
2011-02-13 00:53:58 +08:00
{ ok , User } = egs_users : read ( NPCGID ) ,
[ 16#00001d00 , FromGID , ( User #users.character ) #characters.name ]
2010-09-19 01:57:55 +08:00
end ,
%% log the message as ascii to the console
[ LogName | _ ] = re : split ( BcastName , " \\ 0 \\ 0 " , [ { return , binary } ] ) ,
[ TmpMessage | _ ] = re : split ( ChatMsg , " \\ 0 \\ 0 " , [ { return , binary } ] ) ,
LogMessage = re : replace ( TmpMessage , " \\ n " , " " , [ global , { return , binary } ] ) ,
log ( " chat from ~s : ~s " , [ [ re : replace ( LogName , " \\ 0 " , " " , [ global , { return , binary } ] ) ] , [ re : replace ( LogMessage , " \\ 0 " , " " , [ global , { return , binary } ] ) ] ] ) ,
%% broadcast
2011-02-13 00:53:58 +08:00
{ ok , List } = egs_users : select ( all ) ,
lists : foreach ( fun ( X ) - > X #users.pid ! { egs , chat , UserGID , BcastTypeID , BcastGID , BcastName , Modifiers , ChatMsg } end , List ) ;
2010-09-19 01:57:55 +08:00
%% @todo There's at least 9 different sets of locations. Handle all of them correctly.
event ( counter_background_locations_request , _ State ) - >
psu_game : send_170c ( ) ;
%% @todo Make sure non-mission counters follow the same loading process.
%% @todo Probably validate the From* values, to not send the player back inside a mission.
2010-09-22 06:18:00 +08:00
event ( { counter_enter , CounterID , FromZoneID , FromMapID , FromEntryID } , State = #state { gid = GID } ) - >
2010-09-19 01:57:55 +08:00
log ( " counter load ~b " , [ CounterID ] ) ,
2011-02-13 00:53:58 +08:00
{ ok , OldUser } = egs_users : read ( GID ) ,
2011-02-13 06:28:56 +08:00
FromArea = { element ( 1 , OldUser #users.area ) , FromZoneID , FromMapID } ,
User = OldUser #users { areatype = counter , area = { 16#7fffffff , 0 , 0 } , entryid = 0 , prev_area = FromArea , prev_entryid = FromEntryID } ,
2011-02-13 00:53:58 +08:00
egs_users : write ( User ) ,
2010-11-09 05:03:16 +08:00
QuestData = egs_quests_db : quest ( 0 ) ,
2010-11-28 06:03:04 +08:00
{ ok , ZoneData } = file : read_file ( " data/lobby/counter.zone.nbl " ) ,
2010-09-19 01:57:55 +08:00
%% broadcast unspawn to other people
2011-02-13 00:53:58 +08:00
{ ok , UnspawnList } = egs_users : select ( { neighbors , OldUser } ) ,
lists : foreach ( fun ( Other ) - > Other #users.pid ! { egs , player_unspawn , User } end , UnspawnList ) ,
2010-09-19 01:57:55 +08:00
%% load counter
2010-09-22 10:19:04 +08:00
psu_proto : send_0c00 ( User , State ) ,
2010-11-08 02:45:35 +08:00
psu_proto : send_020e ( QuestData , State ) ,
2010-09-22 10:16:03 +08:00
psu_proto : send_0a05 ( State ) ,
2011-02-13 00:53:58 +08:00
psu_proto : send_010d ( User #users { lid = 0 } , State ) ,
2010-09-26 01:22:13 +08:00
psu_proto : send_0200 ( 0 , mission , State ) ,
2010-11-28 06:03:04 +08:00
psu_proto : send_020f ( ZoneData , 0 , 255 , State ) ,
2010-09-25 22:51:07 +08:00
State2 = State #state { areanb = State #state.areanb + 1 } ,
2011-02-13 00:53:58 +08:00
psu_proto : send_0205 ( User #users { lid = 0 } , 0 , State2 ) ,
2010-09-26 03:46:04 +08:00
psu_proto : send_100e ( CounterID , " Counter " , State2 ) ,
2010-09-25 22:51:07 +08:00
psu_proto : send_0215 ( 0 , State2 ) ,
psu_proto : send_0215 ( 0 , State2 ) ,
psu_proto : send_020c ( State2 ) ,
2010-09-19 01:57:55 +08:00
psu_game : send_1202 ( ) ,
2010-09-26 07:59:40 +08:00
psu_proto : send_1204 ( State2 ) ,
2010-09-19 01:57:55 +08:00
psu_game : send_1206 ( ) ,
psu_game : send_1207 ( ) ,
psu_game : send_1212 ( ) ,
2011-02-13 00:53:58 +08:00
psu_proto : send_0201 ( User #users { lid = 0 } , State2 ) ,
2010-10-02 10:17:23 +08:00
psu_proto : send_0a06 ( User , State2 ) ,
2011-02-13 00:53:58 +08:00
case User #users.partypid of
2010-09-19 01:57:55 +08:00
undefined - > ignore ;
_ - > psu_game : send_022c ( 0 , 16#12 )
end ,
2010-09-25 22:51:07 +08:00
State3 = State2 #state { areanb = State2 #state.areanb + 1 } ,
psu_proto : send_0208 ( State3 ) ,
2010-09-26 01:29:47 +08:00
psu_proto : send_0236 ( State3 ) ,
2010-09-25 22:51:07 +08:00
{ ok , State3 } ;
2010-09-19 01:57:55 +08:00
2010-10-22 05:55:01 +08:00
%% @todo Handle parties to join.
event ( counter_join_party_request , State ) - >
psu_proto : send_1701 ( State ) ;
2010-09-19 01:57:55 +08:00
%% @doc Leave mission counter handler.
event ( counter_leave , State = #state { gid = GID } ) - >
2011-02-13 00:53:58 +08:00
{ ok , User } = egs_users : read ( GID ) ,
PrevArea = User #users.prev_area ,
2011-02-13 06:28:56 +08:00
event ( { area_change , element ( 1 , PrevArea ) , element ( 2 , PrevArea ) , element ( 3 , PrevArea ) , User #users.prev_entryid } , State ) ;
2010-09-19 01:57:55 +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.
2010-10-14 02:09:34 +08:00
%% @todo Rename to counter_bg_request.
2010-10-19 06:20:44 +08:00
event ( { counter_options_request , CounterID } , State ) - >
2010-09-19 01:57:55 +08:00
log ( " counter options request ~p " , [ CounterID ] ) ,
2010-11-06 10:24:55 +08:00
psu_proto : send_1711 ( egs_counters_db : bg ( CounterID ) , State ) ;
2010-09-19 01:57:55 +08:00
%% @todo Handle when the party already exists! And stop doing it wrong.
event ( counter_party_info_request , #state { gid = GID } ) - >
2011-02-13 00:53:58 +08:00
{ ok , User } = egs_users : read ( GID ) ,
psu_game : send_1706 ( ( User #users.character ) #characters.name ) ;
2010-09-19 01:57:55 +08:00
%% @todo Item distribution is always set to random for now.
event ( counter_party_options_request , _ State ) - >
psu_game : send_170a ( ) ;
%% @doc Request the counter's quest files.
2010-10-14 02:09:34 +08:00
event ( { counter_quest_files_request , CounterID } , State ) - >
2010-09-19 01:57:55 +08:00
log ( " counter quest files request ~p " , [ CounterID ] ) ,
2010-11-06 10:24:55 +08:00
psu_proto : send_0c06 ( egs_counters_db : pack ( CounterID ) , State ) ;
2010-09-19 01:57:55 +08:00
%% @doc Counter available mission list request handler.
2010-10-14 02:09:34 +08:00
event ( { counter_quest_options_request , CounterID } , State ) - >
2010-09-19 01:57:55 +08:00
log ( " counter quest options request ~p " , [ CounterID ] ) ,
2010-11-06 10:24:55 +08:00
psu_proto : send_0c10 ( egs_counters_db : opts ( CounterID ) , State ) ;
2010-09-19 01:57:55 +08:00
%% @todo A and B are mostly unknown. Like most of everything else from the command 0e00...
event ( { hit , FromTargetID , ToTargetID , A , B } , State = #state { gid = GID } ) - >
2011-02-13 00:53:58 +08:00
{ ok , User } = egs_users : read ( GID ) ,
2010-09-19 01:57:55 +08:00
%% hit!
#hit_response { type = Type , user = NewUser , exp = HasEXP , damage = Damage , targethp = TargetHP , targetse = TargetSE , events = Events } = psu_instance : hit ( User , FromTargetID , ToTargetID ) ,
case Type of
box - >
%% @todo also has a hit sent, we should send it too
events ( Events , State ) ;
_ - >
2011-02-13 00:53:58 +08:00
PlayerHP = ( NewUser #users.character ) #characters.currenthp ,
2010-09-19 01:57:55 +08:00
case lists : member ( death , TargetSE ) of
true - > SE = 16#01000200 ;
false - > SE = 16#01000000
end ,
psu_game : 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 - >
2011-02-13 00:53:58 +08:00
psu_proto : send_0115 ( NewUser #users { lid = 0 } , ToTargetID , State ) ;
2010-09-19 01:57:55 +08:00
true - > ignore
end ,
%% save
2011-02-13 00:53:58 +08:00
egs_users : write ( NewUser ) ;
2010-09-19 01:57:55 +08:00
event ( { hits , Hits } , State ) - >
events ( Hits , State ) ;
2010-09-25 09:27:07 +08:00
event ( { item_description_request , ItemID } , State ) - >
2010-10-08 23:33:49 +08:00
psu_proto : send_0a11 ( ItemID , egs_items_db : desc ( ItemID ) , State ) ;
2010-09-19 01:57:55 +08:00
%% @todo A and B are unknown.
%% Melee uses a format similar to: AAAA--BBCCCC----DDDDDDDDEE----FF with
%% AAAA the attack sound effect, BB the range, CCCC and DDDDDDDD unknown but related to angular range or similar, EE number of targets and FF the model.
%% Bullets and tech weapons formats are unknown but likely use a slightly different format.
%% @todo Others probably want to see that you changed your weapon.
%% @todo Apparently B is always ItemID+1. Not sure why.
%% @todo Currently use a separate file for the data sent for the weapons.
%% @todo TargetGID and TargetLID must be validated, they're either the player's or his NPC characters.
%% @todo Handle NPC characters properly.
event ( { item_equip , ItemIndex , TargetGID , TargetLID , A , B } , #state { gid = GID } ) - >
2011-02-13 00:53:58 +08:00
case egs_users : item_nth ( GID , ItemIndex ) of
2010-09-19 01:57:55 +08:00
{ ItemID , Variables } when element ( 1 , Variables ) =:= psu_special_item_variables - >
< < Category : 8 , _ : 24 > > = < < ItemID : 32 > > ,
psu_game : send ( < < 16#01050300 : 32 , 0 : 64 , TargetGID : 32 / little , 0 : 64 , 16#00011300 : 32 , GID : 32 / little , 0 : 64 ,
TargetGID : 32 / little , TargetLID : 32 / little , ItemIndex : 8 , 1 : 8 , Category : 8 , A : 8 , B : 32 / little > > ) ;
{ ItemID , Variables } when element ( 1 , Variables ) =:= psu_striking_weapon_item_variables - >
2010-10-08 23:30:24 +08:00
#psu_item { data = Constants } = egs_items_db : read ( ItemID ) ,
2010-09-19 01:57:55 +08:00
#psu_striking_weapon_item { attack_sound = Sound , hitbox_a = HitboxA , hitbox_b = HitboxB ,
hitbox_c = HitboxC , hitbox_d = HitboxD , nb_targets = NbTargets , effect = Effect , model = Model } = Constants ,
< < Category : 8 , _ : 24 > > = < < ItemID : 32 > > ,
{ SoundInt , SoundType } = case Sound of
{ default , Val } - > { Val , 0 } ;
{ custom , Val } - > { Val , 8 }
end ,
psu_game : send ( < < 16#01050300 : 32 , 0 : 64 , TargetGID : 32 / little , 0 : 64 , 16#00011300 : 32 , GID : 32 / little , 0 : 64 ,
TargetGID : 32 / little , TargetLID : 32 / little , ItemIndex : 8 , 1 : 8 , Category : 8 , A : 8 , B : 32 / little ,
SoundInt : 32 / little , HitboxA : 16 , HitboxB : 16 , HitboxC : 16 , HitboxD : 16 , SoundType : 4 , NbTargets : 4 , 0 : 8 , Effect : 8 , Model : 8 > > ) ;
2010-09-29 05:06:27 +08:00
{ ItemID , Variables } when element ( 1 , Variables ) =:= psu_trap_item_variables - >
2010-10-08 23:30:24 +08:00
#psu_item { data = #psu_trap_item { effect = Effect , type = Type } } = egs_items_db : read ( ItemID ) ,
2010-09-29 05:06:27 +08:00
< < Category : 8 , _ : 24 > > = < < ItemID : 32 > > ,
Bin = case Type of
damage - > < < Effect : 8 , 16#0c0a05 : 24 , 16#20140500 : 32 , 16#0001c800 : 32 , 16#10000000 : 32 > > ;
damage_g - > < < Effect : 8 , 16#2c0505 : 24 , 16#0c000600 : 32 , 16#00049001 : 32 , 16#10000000 : 32 > > ;
trap - > < < Effect : 8 , 16#0d0a05 : 24 , 16#61140000 : 32 , 16#0001c800 : 32 , 16#10000000 : 32 > > ;
trap_g - > < < Effect : 8 , 16#4d0505 : 24 , 16#4d000000 : 32 , 16#00049001 : 32 , 16#10000000 : 32 > > ;
trap_ex - > < < Effect : 8 , 16#490a05 : 24 , 16#4500000f : 32 , 16#4b055802 : 32 , 16#10000000 : 32 > >
end ,
psu_game : send ( < < 16#01050300 : 32 , 0 : 64 , TargetGID : 32 / little , 0 : 64 , 16#00011300 : 32 , GID : 32 / little , 0 : 64 ,
TargetGID : 32 / little , TargetLID : 32 / little , ItemIndex : 8 , 1 : 8 , Category : 8 , A : 8 , B : 32 / little , Bin / binary > > ) ;
2010-09-19 01:57:55 +08:00
undefined - >
%% @todo Shouldn't be needed later when NPCs are handled correctly.
ignore
end ;
2010-09-29 05:06:27 +08:00
event ( { item_set_trap , ItemIndex , TargetGID , TargetLID , A , B } , #state { gid = GID } ) - >
2011-02-13 00:53:58 +08:00
{ ItemID , _ Variables } = egs_users : item_nth ( GID , ItemIndex ) ,
egs_users : item_qty_add ( GID , ItemIndex , - 1 ) ,
2010-09-29 05:06:27 +08:00
< < Category : 8 , _ : 24 > > = < < ItemID : 32 > > ,
psu_game : send ( < < 16#01050300 : 32 , 0 : 64 , TargetGID : 32 / little , 0 : 64 , 16#00011300 : 32 , GID : 32 / little , 0 : 64 ,
TargetGID : 32 / little , TargetLID : 32 / little , ItemIndex : 8 , 9 : 8 , Category : 8 , A : 8 , B : 32 / little > > ) ;
2010-09-19 01:57:55 +08:00
%% @todo A and B are unknown.
%% @see item_equip
event ( { item_unequip , ItemIndex , TargetGID , TargetLID , A , B } , #state { gid = GID } ) - >
Category = case ItemIndex of
% units would be 8, traps would be 12
19 - > 2 ; % armor
Y when Y =:= 5 ; Y =:= 6 ; Y =:= 7 - > 0 ; % clothes
_ - > 1 % weapons
end ,
psu_game : send ( < < 16#01050300 : 32 , 0 : 64 , GID : 32 / little - unsigned - integer , 0 : 64 , 16#00011300 : 32 , GID : 32 / little - unsigned - integer ,
0 : 64 , TargetGID : 32 / little - unsigned - integer , TargetLID : 32 / little - unsigned - integer , ItemIndex , 2 , Category , A , B : 32 / little - unsigned - integer > > ) ;
%% @todo Just ignore the meseta price for now and send the player where he wanna be!
2010-10-22 04:56:18 +08:00
event ( lobby_transport_request , State ) - >
psu_proto : send_0c08 ( State ) ;
2010-09-19 01:57:55 +08:00
2010-10-08 09:45:01 +08:00
event ( lumilass_options_request , State = #state { gid = GID } ) - >
2011-02-13 00:53:58 +08:00
{ ok , User } = egs_users : read ( GID ) ,
2010-10-08 09:45:01 +08:00
psu_proto : send_1a03 ( User , State ) ;
2010-09-19 01:57:55 +08:00
%% @todo Probably replenish the player HP when entering a non-mission area rather than when aborting the mission?
event ( mission_abort , State = #state { gid = GID } ) - >
2010-09-24 01:55:03 +08:00
psu_proto : send_1006 ( 11 , State ) ,
2011-02-13 00:53:58 +08:00
{ ok , User } = egs_users : read ( GID ) ,
2010-09-19 01:57:55 +08:00
%% delete the mission
2011-02-13 00:53:58 +08:00
if User #users.instancepid =:= undefined - > ignore ;
true - > psu_instance : stop ( User #users.instancepid )
2010-09-19 01:57:55 +08:00
end ,
%% full hp
2011-02-13 00:53:58 +08:00
Character = User #users.character ,
2010-09-19 01:57:55 +08:00
MaxHP = Character #characters.maxhp ,
NewCharacter = Character #characters { currenthp = MaxHP } ,
2011-02-13 00:53:58 +08:00
NewUser = User #users { character = NewCharacter , setid = 0 , instancepid = undefined } ,
egs_users : write ( NewUser ) ,
2010-09-19 01:57:55 +08:00
%% map change
2011-02-13 00:53:58 +08:00
if User #users.areatype =:= mission - >
PrevArea = User #users.prev_area ,
2011-02-13 06:28:56 +08:00
event ( { area_change , element ( 1 , PrevArea ) , element ( 2 , PrevArea ) , element ( 3 , PrevArea ) , User #users.prev_entryid } , State ) ;
2010-09-19 01:57:55 +08:00
true - > ignore
end ;
%% @todo Forward the mission start to other players of the same party, whatever their location is.
2010-09-24 06:06:14 +08:00
event ( { mission_start , QuestID } , State ) - >
2010-09-19 01:57:55 +08:00
log ( " mission start ~b " , [ QuestID ] ) ,
2010-09-24 06:06:14 +08:00
psu_proto : send_1020 ( State ) ,
2010-09-19 01:57:55 +08:00
psu_game : send_1015 ( QuestID ) ,
psu_game : send_0c02 ( ) ;
%% @doc Force the invite of an NPC character while inside a mission. Mostly used by story missions.
%% Note that the NPC is often removed and reinvited between block/cutscenes.
2010-09-22 06:18:00 +08:00
event ( { npc_force_invite , NPCid } , State = #state { gid = GID } ) - >
2011-02-13 00:53:58 +08:00
{ ok , User } = egs_users : read ( GID ) ,
2010-09-19 01:57:55 +08:00
%% Create NPC.
log ( " npc force invite ~p " , [ NPCid ] ) ,
2011-02-13 00:53:58 +08:00
TmpNPCUser = egs_npc_db : create ( NPCid , ( ( User #users.character ) #characters.mainlevel ) #level.number ) ,
2010-09-19 01:57:55 +08:00
%% Create and join party.
2011-02-13 00:53:58 +08:00
case User #users.partypid of
2010-09-19 01:57:55 +08:00
undefined - >
{ ok , PartyPid } = psu_party : start_link ( GID ) ;
PartyPid - >
ignore
end ,
2011-02-13 07:35:05 +08:00
{ ok , PartyPos } = psu_party : join ( PartyPid , npc , TmpNPCUser #users.gid ) ,
2011-02-13 00:53:58 +08:00
#users { instancepid = InstancePid , area = Area , entryid = EntryID , pos = Pos } = User ,
NPCUser = TmpNPCUser #users { lid = PartyPos , partypid = PartyPid , instancepid = InstancePid , areatype = mission , area = Area , entryid = EntryID , pos = Pos } ,
egs_users : write ( NPCUser ) ,
egs_users : write ( User #users { partypid = PartyPid } ) ,
2010-09-19 01:57:55 +08:00
%% Send stuff.
2011-02-13 00:53:58 +08:00
Character = NPCUser #users.character ,
2010-09-19 01:57:55 +08:00
SentNPCCharacter = Character #characters { gid = NPCid , npcid = NPCid } ,
2011-02-13 00:53:58 +08:00
SentNPCUser = NPCUser #users { character = SentNPCCharacter } ,
2010-09-22 06:18:00 +08:00
psu_proto : send_010d ( SentNPCUser , State ) ,
2010-09-22 06:21:25 +08:00
psu_proto : send_0201 ( SentNPCUser , State ) ,
2010-09-22 09:44:49 +08:00
psu_proto : send_0215 ( 0 , State ) ,
2011-02-13 07:35:05 +08:00
psu_game : send_0a04 ( SentNPCUser #users.gid ) ,
2010-09-19 01:57:55 +08:00
psu_game : send_022c ( 0 , 16#12 ) ,
psu_game : send_1004 ( npc_mission , SentNPCUser , PartyPos ) ,
2011-02-13 00:53:58 +08:00
psu_game : send_100f ( ( SentNPCUser #users.character ) #characters.npcid , PartyPos ) ,
2010-09-19 01:57:55 +08:00
psu_game : send_1601 ( PartyPos ) ;
%% @todo Also at the end send a 101a (NPC:16, PartyPos:16, ffffffff). Not sure about PartyPos.
event ( { npc_invite , NPCid } , #state { gid = GID } ) - >
2011-02-13 00:53:58 +08:00
{ ok , User } = egs_users : read ( GID ) ,
2010-09-19 01:57:55 +08:00
%% Create NPC.
log ( " invited npcid ~b " , [ NPCid ] ) ,
2011-02-13 00:53:58 +08:00
TmpNPCUser = egs_npc_db : create ( NPCid , ( ( User #users.character ) #characters.mainlevel ) #level.number ) ,
2010-09-19 01:57:55 +08:00
%% Create and join party.
2011-02-13 00:53:58 +08:00
case User #users.partypid of
2010-09-19 01:57:55 +08:00
undefined - >
{ ok , PartyPid } = psu_party : start_link ( GID ) ,
psu_game : send_022c ( 0 , 16#12 ) ;
PartyPid - >
ignore
end ,
2011-02-13 07:35:05 +08:00
{ ok , PartyPos } = psu_party : join ( PartyPid , npc , TmpNPCUser #users.gid ) ,
2011-02-13 00:53:58 +08:00
NPCUser = TmpNPCUser #users { lid = PartyPos , partypid = PartyPid } ,
egs_users : write ( NPCUser ) ,
egs_users : write ( User #users { partypid = PartyPid } ) ,
2010-09-19 01:57:55 +08:00
%% Send stuff.
2011-02-13 00:53:58 +08:00
Character = NPCUser #users.character ,
2010-09-19 01:57:55 +08:00
SentNPCCharacter = Character #characters { gid = NPCid , npcid = NPCid } ,
2011-02-13 00:53:58 +08:00
SentNPCUser = NPCUser #users { character = SentNPCCharacter } ,
2010-09-19 01:57:55 +08:00
psu_game : send_1004 ( npc_invite , SentNPCUser , PartyPos ) ,
psu_game : send_101a ( NPCid , PartyPos ) ;
%% @todo Should be 0115(money) 010a03(confirm sale).
2010-09-30 06:32:32 +08:00
event ( { npc_shop_buy , ShopItemIndex , QuantityOrColor } , State = #state { gid = GID } ) - >
2011-02-13 00:53:58 +08:00
ShopID = egs_users : shop_get ( GID ) ,
2010-10-09 02:56:48 +08:00
ItemID = egs_shops_db : nth ( ShopID , ShopItemIndex + 1 ) ,
2010-09-30 06:32:32 +08:00
log ( " npc shop ~p buy itemid ~8.16.0b quantity/color+1 ~p " , [ ShopID , ItemID , QuantityOrColor ] ) ,
2010-10-08 23:30:24 +08:00
#psu_item { name = Name , rarity = Rarity , buy_price = BuyPrice , sell_price = SellPrice , data = Constants } = egs_items_db : read ( ItemID ) ,
2010-10-09 07:09:53 +08:00
{ Quantity , Variables } = case element ( 1 , Constants ) of
2010-09-30 06:32:32 +08:00
psu_clothing_item - >
if QuantityOrColor > = 1 , QuantityOrColor =< 10 - >
2010-10-09 07:09:53 +08:00
{ 1 , #psu_clothing_item_variables { color = QuantityOrColor - 1 } }
2010-09-30 06:32:32 +08:00
end ;
psu_consumable_item - >
2010-10-09 07:09:53 +08:00
{ QuantityOrColor , #psu_consumable_item_variables { quantity = QuantityOrColor } } ;
2010-09-30 06:32:32 +08:00
psu_parts_item - >
2010-10-09 07:09:53 +08:00
{ 1 , #psu_parts_item_variables { } } ;
2010-09-30 06:32:32 +08:00
psu_special_item - >
2010-10-09 07:09:53 +08:00
{ 1 , #psu_special_item_variables { } } ;
2010-09-30 06:32:32 +08:00
psu_striking_weapon_item - >
#psu_striking_weapon_item { pp = PP , shop_element = Element } = Constants ,
2010-10-09 07:09:53 +08:00
{ 1 , #psu_striking_weapon_item_variables { current_pp = PP , max_pp = PP , element = Element } } ;
2010-09-30 06:32:32 +08:00
psu_trap_item - >
2010-10-09 07:09:53 +08:00
{ QuantityOrColor , #psu_trap_item_variables { quantity = QuantityOrColor } }
2010-09-30 06:32:32 +08:00
end ,
2011-02-13 00:53:58 +08:00
egs_users : money_add ( GID , - 1 * BuyPrice * Quantity ) ,
ItemUUID = egs_users : item_add ( GID , ItemID , Variables ) ,
{ ok , User } = egs_users : read ( GID ) ,
psu_proto : send_0115 ( User #users { lid = 0 } , State ) , %% @todo This one is apparently broadcast to everyone in the same zone.
2010-09-30 06:32:32 +08:00
%% @todo Following command isn't done 100% properly.
UCS2Name = < < < < X : 8 , 0 : 8 > > | | X < - Name > > ,
NamePadding = 8 * ( 46 - byte_size ( UCS2Name ) ) ,
< < Category : 8 , _ : 24 > > = < < ItemID : 32 > > ,
RarityInt = Rarity - 1 ,
psu_game : send ( < < 16#010a0300 : 32 , 0 : 64 , GID : 32 / little , 0 : 64 , 16#00011300 : 32 , GID : 32 / little , 0 : 64 ,
GID : 32 / little , 0 : 32 , 2 : 16 / little , 0 : 16 , ( psu_game : build_item_variables ( ItemID , ItemUUID , Variables ) ) / binary ,
UCS2Name / binary , 0 : NamePadding , RarityInt : 8 , Category : 8 , SellPrice : 32 / little , ( psu_game : build_item_constants ( Constants ) ) / binary > > ) ;
2010-09-19 01:57:55 +08:00
%% @todo Currently send the normal items shop for all shops, differentiate.
event ( { npc_shop_enter , ShopID } , #state { gid = GID } ) - >
log ( " npc shop enter ~p " , [ ShopID ] ) ,
2011-02-13 00:53:58 +08:00
egs_users : shop_enter ( GID , ShopID ) ,
2010-10-09 02:56:48 +08:00
psu_game : send_010a ( egs_shops_db : read ( ShopID ) ) ;
2010-09-19 01:57:55 +08:00
event ( { npc_shop_leave , ShopID } , #state { gid = GID } ) - >
log ( " npc shop leave ~p " , [ ShopID ] ) ,
2011-02-13 00:53:58 +08:00
egs_users : shop_leave ( GID ) ,
2010-09-19 01:57:55 +08:00
psu_game : send ( < < 16#010a0300 : 32 , 0 : 64 , GID : 32 / little - unsigned - integer , 0 : 64 , 16#00011300 : 32 ,
GID : 32 / little - unsigned - integer , 0 : 64 , GID : 32 / little - unsigned - integer , 0 : 32 > > ) ;
%% @todo Should be 0115(money) 010a03(confirm sale).
event ( { npc_shop_sell , InventoryItemIndex , Quantity } , _ State ) - >
log ( " npc shop sell itemindex ~p quantity ~p " , [ InventoryItemIndex , Quantity ] ) ;
%% @todo First 1a02 value should be non-0.
%% @todo Could the 2nd 1a02 parameter simply be the shop type or something?
%% @todo Although the values replied should be right, they seem mostly ignored by the client.
2010-10-06 23:28:55 +08:00
event ( { npc_shop_request , ShopID } , State ) - >
2010-09-19 01:57:55 +08:00
log ( " npc shop request ~p " , [ ShopID ] ) ,
case ShopID of
2010-10-06 23:28:55 +08:00
80 - > psu_proto : send_1a02 ( 17 , 17 , 3 , 9 , State ) ; %% lumilass
90 - > psu_proto : send_1a02 ( 5 , 1 , 4 , 5 , State ) ; %% parum weapon grinding
91 - > psu_proto : send_1a02 ( 5 , 5 , 4 , 7 , State ) ; %% tenora weapon grinding
92 - > psu_proto : send_1a02 ( 5 , 8 , 4 , 0 , State ) ; %% yohmei weapon grinding
93 - > psu_proto : send_1a02 ( 5 , 18 , 4 , 0 , State ) ; %% kubara weapon grinding
_ - > psu_proto : send_1a02 ( 0 , 1 , 0 , 0 , State )
2010-09-19 01:57:55 +08:00
end ;
%% @todo Not sure what are those hardcoded values.
event ( { object_boss_gate_activate , ObjectID } , _ State ) - >
psu_game : send_1213 ( ObjectID , 0 ) ,
psu_game : send_1215 ( 2 , 16#7008 ) ,
%% @todo Following sent after the warp?
psu_game : send_1213 ( 37 , 0 ) ,
%% @todo Why resend this?
psu_game : send_1213 ( ObjectID , 0 ) ;
event ( { object_boss_gate_enter , ObjectID } , _ State ) - >
psu_game : send_1213 ( ObjectID , 1 ) ;
%% @todo Do we need to send something back here?
event ( { object_boss_gate_leave , _ ObjectID } , _ State ) - >
ignore ;
event ( { object_box_destroy , ObjectID } , _ State ) - >
psu_game : send_1213 ( ObjectID , 3 ) ;
2011-02-13 00:53:58 +08:00
%% @todo Second send_1211 argument should be User#users.lid. Fix when it's correctly handled.
2010-09-19 01:57:55 +08:00
event ( { object_chair_sit , ObjectTargetID } , _ State ) - >
2011-02-13 00:53:58 +08:00
%~ {ok, User} = egs_users:read(get(gid)),
2010-09-19 01:57:55 +08:00
psu_game : send_1211 ( ObjectTargetID , 0 , 8 , 0 ) ;
2011-02-13 00:53:58 +08:00
%% @todo Second psu_game:send_1211 argument should be User#users.lid. Fix when it's correctly handled.
2010-09-19 01:57:55 +08:00
event ( { object_chair_stand , ObjectTargetID } , _ State ) - >
2011-02-13 00:53:58 +08:00
%~ {ok, User} = egs_users:read(get(gid)),
2010-09-19 01:57:55 +08:00
psu_game : send_1211 ( ObjectTargetID , 0 , 8 , 2 ) ;
event ( { object_crystal_activate , ObjectID } , _ State ) - >
psu_game : send_1213 ( ObjectID , 1 ) ;
%% @doc Server-side event.
event ( { object_event_trigger , BlockID , EventID } , _ State ) - >
psu_game : send_1205 ( EventID , BlockID , 0 ) ;
event ( { object_goggle_target_activate , ObjectID } , #state { gid = GID } ) - >
2011-02-13 00:53:58 +08:00
{ ok , User } = egs_users : read ( GID ) ,
2011-02-13 06:28:56 +08:00
{ BlockID , EventID } = psu_instance : std_event ( User #users.instancepid , element ( 2 , User #users.area ) , ObjectID ) ,
2010-09-19 01:57:55 +08:00
psu_game : send_1205 ( EventID , BlockID , 0 ) ,
psu_game : send_1213 ( ObjectID , 8 ) ;
2010-09-24 08:33:22 +08:00
%% @todo Make NPC characters heal too.
event ( { object_healing_pad_tick , [ _ PartyPos ] } , State = #state { gid = GID } ) - >
2011-02-13 00:53:58 +08:00
{ ok , User } = egs_users : read ( GID ) ,
Character = User #users.character ,
2010-09-24 08:33:22 +08:00
if Character #characters.currenthp =:= Character #characters.maxhp - > ignore ;
true - >
NewHP = Character #characters.currenthp + Character #characters.maxhp div 10 ,
NewHP2 = if NewHP > Character #characters.maxhp - > Character #characters.maxhp ; true - > NewHP end ,
2011-02-13 00:53:58 +08:00
User2 = User #users { character = Character #characters { currenthp = NewHP2 } } ,
egs_users : write ( User2 ) ,
psu_proto : send_0117 ( User2 #users { lid = 0 } , State ) ,
psu_proto : send_0111 ( User2 #users { lid = 0 } , 4 , State )
2010-09-24 08:33:22 +08:00
end ;
2010-09-19 01:57:55 +08:00
event ( { object_key_console_enable , ObjectID } , #state { gid = GID } ) - >
2011-02-13 00:53:58 +08:00
{ ok , User } = egs_users : read ( GID ) ,
2011-02-13 06:28:56 +08:00
{ BlockID , [ EventID | _ ] } = psu_instance : std_event ( User #users.instancepid , element ( 2 , User #users.area ) , ObjectID ) ,
2010-09-19 01:57:55 +08:00
psu_game : send_1205 ( EventID , BlockID , 0 ) ,
psu_game : send_1213 ( ObjectID , 1 ) ;
event ( { object_key_console_init , ObjectID } , #state { gid = GID } ) - >
2011-02-13 00:53:58 +08:00
{ ok , User } = egs_users : read ( GID ) ,
2011-02-13 06:28:56 +08:00
{ BlockID , [ _ , EventID , _ ] } = psu_instance : std_event ( User #users.instancepid , element ( 2 , User #users.area ) , ObjectID ) ,
2010-09-19 01:57:55 +08:00
psu_game : send_1205 ( EventID , BlockID , 0 ) ;
event ( { object_key_console_open_gate , ObjectID } , #state { gid = GID } ) - >
2011-02-13 00:53:58 +08:00
{ ok , User } = egs_users : read ( GID ) ,
2011-02-13 06:28:56 +08:00
{ BlockID , [ _ , _ , EventID ] } = psu_instance : std_event ( User #users.instancepid , element ( 2 , User #users.area ) , ObjectID ) ,
2010-09-19 01:57:55 +08:00
psu_game : send_1205 ( EventID , BlockID , 0 ) ,
psu_game : 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 } , #state { gid = GID } ) - >
2011-02-13 00:53:58 +08:00
{ ok , User } = egs_users : read ( GID ) ,
2011-02-13 06:28:56 +08:00
{ BlockID , [ EventID | _ ] } = psu_instance : std_event ( User #users.instancepid , element ( 2 , User #users.area ) , ObjectID ) ,
2010-09-19 01:57:55 +08:00
psu_game : send_1205 ( EventID , BlockID , 0 ) ,
psu_game : send_1213 ( ObjectID , 1 ) ;
%% @todo Some switch objects apparently work differently, like the light switch in Mines in MAG'.
event ( { object_switch_off , ObjectID } , #state { gid = GID } ) - >
2011-02-13 00:53:58 +08:00
{ ok , User } = egs_users : read ( GID ) ,
2011-02-13 06:28:56 +08:00
{ BlockID , EventID } = psu_instance : std_event ( User #users.instancepid , element ( 2 , User #users.area ) , ObjectID ) ,
2010-09-19 01:57:55 +08:00
psu_game : send_1205 ( EventID , BlockID , 1 ) ,
psu_game : send_1213 ( ObjectID , 0 ) ;
event ( { object_switch_on , ObjectID } , #state { gid = GID } ) - >
2011-02-13 00:53:58 +08:00
{ ok , User } = egs_users : read ( GID ) ,
2011-02-13 06:28:56 +08:00
{ BlockID , EventID } = psu_instance : std_event ( User #users.instancepid , element ( 2 , User #users.area ) , ObjectID ) ,
2010-09-19 01:57:55 +08:00
psu_game : send_1205 ( EventID , BlockID , 0 ) ,
psu_game : send_1213 ( ObjectID , 1 ) ;
event ( { object_vehicle_boost_enable , ObjectID } , _ State ) - >
psu_game : send_1213 ( ObjectID , 1 ) ;
event ( { object_vehicle_boost_respawn , ObjectID } , _ State ) - >
psu_game : send_1213 ( ObjectID , 0 ) ;
2011-02-13 00:53:58 +08:00
%% @todo Second send_1211 argument should be User#users.lid. Fix when it's correctly handled.
2010-09-19 01:57:55 +08:00
event ( { object_warp_take , BlockID , ListNb , ObjectNb } , #state { gid = GID } ) - >
2011-02-13 00:53:58 +08:00
{ ok , User } = egs_users : read ( GID ) ,
2011-02-13 06:28:56 +08:00
Pos = psu_instance : warp_event ( User #users.instancepid , element ( 2 , User #users.area ) , BlockID , ListNb , ObjectNb ) ,
2011-02-13 00:53:58 +08:00
NewUser = User #users { pos = Pos } ,
egs_users : write ( NewUser ) ,
psu_game : send_0503 ( User #users.pos ) ,
2010-09-19 01:57:55 +08:00
psu_game : send_1211 ( 16#ffffffff , 0 , 14 , 0 ) ;
2010-10-22 04:31:06 +08:00
%% @todo Don't send_0204 if the player is removed from the party while in the lobby I guess.
2010-09-22 09:44:49 +08:00
event ( { party_remove_member , PartyPos } , State = #state { gid = GID } ) - >
2010-09-19 01:57:55 +08:00
log ( " party remove member ~b " , [ PartyPos ] ) ,
2011-02-13 00:53:58 +08:00
{ ok , DestUser } = egs_users : read ( GID ) ,
{ ok , RemovedGID } = psu_party : get_member ( DestUser #users.partypid , PartyPos ) ,
psu_party : remove_member ( DestUser #users.partypid , PartyPos ) ,
{ ok , RemovedUser } = egs_users : read ( RemovedGID ) ,
case ( RemovedUser #users.character ) #characters.type of
npc - > egs_users : delete ( RemovedGID ) ;
2010-09-19 01:57:55 +08:00
_ - > ignore
end ,
2010-09-24 01:55:03 +08:00
psu_proto : send_1006 ( 8 , PartyPos , State ) ,
2010-10-22 04:31:06 +08:00
psu_proto : send_0204 ( RemovedUser , State ) ,
2010-09-22 09:44:49 +08:00
psu_proto : send_0215 ( 0 , State ) ;
2010-09-19 01:57:55 +08:00
2010-10-11 09:20:20 +08:00
event ( { player_options_change , Options } , #state { gid = GID , slot = Slot } ) - >
Folder = egs_accounts : get_folder ( GID ) ,
file : write_file ( io_lib : format ( " save/ ~s / ~b -character.options " , [ Folder , Slot ] ) , Options ) ;
2010-09-19 01:57:55 +08:00
%% @todo If the player has a scape, use it! Otherwise red screen.
2010-09-24 00:30:50 +08:00
%% @todo Right now we force revive with a dummy HP value.
event ( player_death , State = #state { gid = GID } ) - >
2010-09-19 01:57:55 +08:00
% @todo send_0115(get(gid), 16#ffffffff, LV=1, EXP=idk, Money=1000), % apparently sent everytime you die...
%% use scape:
NewHP = 10 ,
2011-02-13 00:53:58 +08:00
{ ok , User } = egs_users : read ( GID ) ,
Char = User #users.character ,
User2 = User #users { character = Char #characters { currenthp = NewHP } } ,
egs_users : write ( User2 ) ,
psu_proto : send_0117 ( User2 #users { lid = 0 } , State ) ,
2010-09-24 05:16:24 +08:00
psu_proto : send_1022 ( User2 , State ) ;
2010-09-19 01:57:55 +08:00
%% red screen with return to lobby choice:
2010-09-24 03:23:59 +08:00
%~ psu_proto:send_0111(User2, 3, 1, State);
2010-09-19 01:57:55 +08:00
%% @todo Refill the player's HP to maximum, remove SEs etc.
event ( player_death_return_to_lobby , State = #state { gid = GID } ) - >
2011-02-13 00:53:58 +08:00
{ ok , User } = egs_users : read ( GID ) ,
PrevArea = User #users.prev_area ,
2011-02-13 06:28:56 +08:00
event ( { area_change , element ( 1 , PrevArea ) , element ( 2 , PrevArea ) , element ( 3 , PrevArea ) , User #users.prev_entryid } , State ) ;
2010-09-19 01:57:55 +08:00
2010-09-24 04:33:53 +08:00
event ( player_type_availability_request , State ) - >
psu_proto : send_1a07 ( State ) ;
2010-09-19 01:57:55 +08:00
event ( player_type_capabilities_request , _ State ) - >
psu_game : send_0113 ( ) ;
event ( ppcube_request , _ State ) - >
psu_game : send_1a04 ( ) ;
2010-10-21 23:00:30 +08:00
event ( unicube_request , State ) - >
psu_proto : send_021e ( egs_universes : all ( ) , State ) ;
2010-09-19 01:57:55 +08:00
2010-10-21 23:00:30 +08:00
%% @todo When selecting 'Your room', don't load a default room that's not yours.
event ( { unicube_select , cancel , _ EntryID } , _ State ) - >
ignore ;
2010-10-11 09:20:20 +08:00
event ( { unicube_select , Selection , EntryID } , State = #state { gid = GID } ) - >
2011-02-13 00:53:58 +08:00
{ ok , User } = egs_users : read ( GID ) ,
2010-09-19 01:57:55 +08:00
case Selection of
16#ffffffff - >
2010-10-21 23:00:30 +08:00
UniID = egs_universes : myroomid ( ) ,
2011-02-13 06:28:56 +08:00
User2 = User #users { uni = UniID , area = { 1120000 , 0 , 100 } , entryid = 0 } ;
2010-10-21 23:00:30 +08:00
_ - >
UniID = Selection ,
2011-02-13 00:53:58 +08:00
User2 = User #users { uni = UniID , entryid = EntryID }
2010-10-21 23:00:30 +08:00
end ,
psu_proto : send_0230 ( State ) ,
%% 0220
2011-02-13 00:53:58 +08:00
case User #users.partypid of
2010-10-21 23:00:30 +08:00
undefined - > ignore ;
PartyPid - >
%% @todo Replace stop by leave when leaving stops the party correctly when nobody's there anymore.
2011-02-13 07:35:05 +08:00
%~ psu_party:leave(User#users.partypid, User#users.gid)
2010-10-21 23:00:30 +08:00
{ ok , NPCList } = psu_party : get_npc ( PartyPid ) ,
2011-02-13 00:53:58 +08:00
[ egs_users : delete ( NPCGID ) | | { _ Spot , NPCGID } < - NPCList ] ,
2010-10-21 23:00:30 +08:00
psu_party : stop ( PartyPid )
end ,
2011-02-13 00:53:58 +08:00
egs_users : write ( User2 ) ,
egs_universes : leave ( User #users.uni ) ,
2010-10-21 23:00:30 +08:00
egs_universes : enter ( UniID ) ,
psu_game : char_load ( User2 , State ) .
2010-09-19 01:57:55 +08:00
%% Internal.
%% @doc Trigger many events.
events ( Events , State ) - >
[ event ( Event , State ) | | Event < - Events ] ,
ok .
%% @doc Log message to the console.
log ( Message ) - >
io : format ( " ~p : ~s ~n " , [ get ( gid ) , Message ] ) .
log ( Message , Format ) - >
FormattedMessage = io_lib : format ( Message , Format ) ,
log ( FormattedMessage ) .