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 ] ) .
%% @todo This header is probably only included because of egs_user_model. We don't want that here.
- include ( " include/records.hrl " ) .
- include ( " include/maps.hrl " ) .
- include ( " include/psu/items.hrl " ) .
- record ( state , { socket , gid } ) .
%% @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.
info ( { egs , chat , ChatTypeID , ChatGID , ChatName , ChatModifiers , ChatMessage } , _ State ) - >
psu_game : send_0304 ( ChatTypeID , ChatGID , ChatName , ChatModifiers , ChatMessage ) ;
%% @doc Inform the client that a player has spawn.
%% @todo Should be something along the lines of 010d 0205 203 201.
info ( { egs , player_spawn , _ Player } , #state { gid = GID } ) - >
{ ok , User } = egs_user_model : read ( GID ) ,
{ ok , SpawnList } = egs_user_model : select ( { neighbors , User } ) ,
psu_game : send_0233 ( SpawnList ) ;
%% @doc Inform the client that a player has unspawn.
info ( { egs , player_unspawn , Player } , #state { gid = GID } ) - >
{ ok , User } = egs_user_model : read ( GID ) ,
psu_game : send_0204 ( User , Player , 5 ) ;
%% @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 ,
{ ok , User } = egs_user_model : read ( 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 ) ,
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 ,
{ ok , User } = egs_user_model : read ( 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 ) ,
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 ,
case egs_user_model : read ( GID ) of
{ error , _ Reason } - >
ignore ;
{ ok , Self } - >
LID = Self #egs_user_model.lid ,
Packet = < < A / binary , 16#00011300 : 32 , GID : 32 / little - unsigned - integer , B / binary ,
GID : 32 / little - unsigned - integer , LID : 32 / little - unsigned - integer , C / binary > > ,
{ ok , SpawnList } = egs_user_model : select ( { neighbors , Self } ) ,
lists : foreach ( fun ( User ) - > User #egs_user_model.pid ! { egs , cast , Packet } end , SpawnList )
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 ] ) ,
{ ok , User } = egs_user_model : read ( GID ) ,
{ BlockID , EventID } = psu_instance : spawn_cleared_event ( User #egs_user_model.instancepid , ( User #egs_user_model.area ) #psu_area.zoneid , SpawnID ) ,
if EventID =:= false - > ignore ;
true - > 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?
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.
%% @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 } , State ) - >
event ( { area_change , QuestID , ZoneID , MapID , EntryID , 16#ffffffff } , State ) ;
event ( { area_change , QuestID , ZoneID , MapID , EntryID , PartyPos } , _ State ) - >
case PartyPos of
16#ffffffff - >
log ( " area change ( ~b , ~b , ~b , ~b , ~b ) " , [ QuestID , ZoneID , MapID , EntryID , PartyPos ] ) ,
psu_game : area_load ( QuestID , ZoneID , MapID , EntryID ) ;
_ 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 } ) - >
{ ok , User = #egs_user_model { area = #psu_area { questid = QuestID , zoneid = ZoneID , mapid = MapID } ,
entryid = EntryID } } = egs_user_model : read ( GID ) ,
egs_user_model : write ( User #egs_user_model { area = #psu_area { questid = 0 , zoneid = 0 , mapid = 0 } , entryid = 0 } ) ,
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.
{ ok , User } = egs_user_model : read ( UserGID ) ,
[ 16#00001200 , User #egs_user_model.id , ( User #egs_user_model.character ) #characters.name ] ;
NPCGID - > %% npc chat: @todo Check that the player is the party leader and this npc is in his party.
{ ok , User } = egs_user_model : read ( NPCGID ) ,
[ 16#00001d00 , FromGID , ( User #egs_user_model.character ) #characters.name ]
end ,
%% log the message as ascii to the console
[ LogName | _ ] = re : split ( BcastName , " \\ 0 \\ 0 " , [ { return , binary } ] ) ,
[ TmpMessage | _ ] = re : split ( ChatMsg , " \\ 0 \\ 0 " , [ { return , binary } ] ) ,
LogMessage = re : replace ( TmpMessage , " \\ n " , " " , [ global , { return , binary } ] ) ,
log ( " chat from ~s : ~s " , [ [ re : replace ( LogName , " \\ 0 " , " " , [ global , { return , binary } ] ) ] , [ re : replace ( LogMessage , " \\ 0 " , " " , [ global , { return , binary } ] ) ] ] ) ,
%% broadcast
{ ok , List } = egs_user_model : select ( all ) ,
lists : foreach ( fun ( X ) - > X #egs_user_model.pid ! { psu_chat , BcastTypeID , BcastGID , BcastName , Modifiers , ChatMsg } end , List ) ;
%% @todo There's at least 9 different sets of locations. Handle all of them correctly.
event ( counter_background_locations_request , _ 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.
event ( { counter_enter , CounterID , FromZoneID , FromMapID , FromEntryID } , #state { gid = GID } ) - >
log ( " counter load ~b " , [ CounterID ] ) ,
{ ok , OldUser } = egs_user_model : read ( GID ) ,
OldArea = OldUser #egs_user_model.area ,
FromArea = { psu_area , OldArea #psu_area.questid , FromZoneID , FromMapID } ,
User = OldUser #egs_user_model { areatype = counter , area = { psu_area , 16#7fffffff , 0 , 0 } , entryid = 0 , prev_area = FromArea , prev_entryid = FromEntryID } ,
egs_user_model : write ( User ) ,
AreaName = " Counter " ,
QuestFile = " data/lobby/counter.quest.nbl " ,
ZoneFile = " data/lobby/counter.zone.nbl " ,
%% broadcast unspawn to other people
{ ok , UnspawnList } = egs_user_model : select ( { neighbors , OldUser } ) ,
lists : foreach ( fun ( Other ) - > Other #egs_user_model.pid ! { psu_player_unspawn , User } end , UnspawnList ) ,
%% load counter
psu_proto : send_0c00 ( User ) ,
psu_proto : send_020e ( User , QuestFile ) ,
psu_proto : send_0a05 ( User ) ,
psu_proto : send_010d ( User , User #egs_user_model { lid = 0 } ) ,
psu_game : send_0200 ( mission ) ,
psu_game : send_020f ( ZoneFile , 0 , 16#ff ) ,
psu_proto : send_0205 ( User , 0 ) ,
psu_game : send_100e ( 16#7fffffff , 0 , 0 , AreaName , CounterID ) ,
psu_proto : send_0215 ( User , 0 ) ,
psu_proto : send_0215 ( User , 0 ) ,
psu_proto : send_020c ( User ) ,
psu_game : send_1202 ( ) ,
psu_game : send_1204 ( ) ,
psu_game : send_1206 ( ) ,
psu_game : send_1207 ( ) ,
psu_game : send_1212 ( ) ,
psu_proto : send_0201 ( User , User #egs_user_model { lid = 0 } ) ,
psu_game : send_0a06 ( ) ,
case User #egs_user_model.partypid of
undefined - > ignore ;
_ - > psu_game : send_022c ( 0 , 16#12 )
end ,
psu_game : send_0208 ( ) ,
psu_game : send_0236 ( ) ;
%% @doc Leave mission counter handler.
event ( counter_leave , State = #state { gid = GID } ) - >
{ ok , User } = egs_user_model : read ( GID ) ,
PrevArea = User #egs_user_model.prev_area ,
event ( { area_change , PrevArea #psu_area.questid , PrevArea #psu_area.zoneid , PrevArea #psu_area.mapid , User #egs_user_model.prev_entryid } , State ) ;
%% @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 } , _ State ) - >
log ( " counter options request ~p " , [ CounterID ] ) ,
[ { quests , _ } , { bg , Background } | _ Tail ] = proplists : get_value ( CounterID , ? COUNTERS ) ,
psu_game : send_1711 ( Background ) ;
%% @todo Handle when the party already exists! And stop doing it wrong.
event ( counter_party_info_request , #state { gid = GID } ) - >
{ ok , User } = egs_user_model : read ( GID ) ,
psu_game : send_1706 ( ( User #egs_user_model.character ) #characters.name ) ;
%% @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.
event ( { counter_quest_files_request , CounterID } , _ State ) - >
log ( " counter quest files request ~p " , [ CounterID ] ) ,
[ { quests , Filename } | _ Tail ] = proplists : get_value ( CounterID , ? COUNTERS ) ,
psu_game : send_0c06 ( Filename ) ;
%% @doc Counter available mission list request handler.
event ( { counter_quest_options_request , CounterID } , _ State ) - >
log ( " counter quest options request ~p " , [ CounterID ] ) ,
[ { quests , _ } , { bg , _ } , { options , Options } ] = proplists : get_value ( CounterID , ? COUNTERS ) ,
psu_game : send_0c10 ( Options ) ;
%% @todo A and B are mostly unknown. Like most of everything else from the command 0e00...
event ( { hit , FromTargetID , ToTargetID , A , B } , State = #state { gid = GID } ) - >
{ ok , User } = egs_user_model : read ( GID ) ,
%% hit!
#hit_response { type = Type , user = NewUser , exp = HasEXP , damage = Damage , targethp = TargetHP , targetse = TargetSE , events = Events } = psu_instance : hit ( User , FromTargetID , ToTargetID ) ,
case Type of
box - >
%% @todo also has a hit sent, we should send it too
events ( Events , State ) ;
_ - >
PlayerHP = ( NewUser #egs_user_model.character ) #characters.currenthp ,
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 - >
Character = NewUser #egs_user_model.character ,
Level = Character #characters.mainlevel ,
psu_game : 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 } , State ) - >
events ( Hits , State ) ;
%% @todo Send something other than just "dammy".
event ( { item_description_request , ItemID } , _ State ) - >
case proplists : get_value ( ItemID , ? ITEMS ) of
undefined - > psu_game : send_0a11 ( ItemID , " Always bet on Dammy. " ) ;
#psu_item { description = Desc } - > psu_game : send_0a11 ( ItemID , Desc )
end ;
%% @todo A and B are unknown.
%% Melee uses a format similar to: AAAA--BBCCCC----DDDDDDDDEE----FF with
%% AAAA the attack sound effect, BB the range, CCCC and DDDDDDDD unknown but related to angular range or similar, EE number of targets and FF the model.
%% Bullets and tech weapons formats are unknown but likely use a slightly different format.
%% @todo Others probably want to see that you changed your weapon.
%% @todo Apparently B is always ItemID+1. Not sure why.
%% @todo Currently use a separate file for the data sent for the weapons.
%% @todo TargetGID and TargetLID must be validated, they're either the player's or his NPC characters.
%% @todo Handle NPC characters properly.
event ( { item_equip , ItemIndex , TargetGID , TargetLID , A , B } , #state { gid = GID } ) - >
{ ok , User } = egs_user_model : read ( GID ) ,
Inventory = ( User #egs_user_model.character ) #characters.inventory ,
case lists : nth ( ItemIndex + 1 , Inventory ) of
{ 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 - >
ItemInfo = proplists : get_value ( ItemID , ? ITEMS ) ,
#psu_item { data = Constants } = ItemInfo ,
#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 > > ) ;
undefined - >
%% @todo Shouldn't be needed later when NPCs are handled correctly.
ignore
end ;
%% @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!
event ( lobby_transport_request , _ State ) - >
psu_game : send_0c08 ( true ) ;
event ( lumilass_options_request , _ State ) - >
psu_game : send_1a03 ( ) ;
%% @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 } ) - >
psu_game : send_1006 ( 11 , 0 ) ,
{ ok , User } = egs_user_model : read ( GID ) ,
%% delete the mission
if User #egs_user_model.instancepid =:= undefined - > ignore ;
true - > psu_instance : stop ( User #egs_user_model.instancepid )
end ,
%% full hp
Character = User #egs_user_model.character ,
MaxHP = Character #characters.maxhp ,
NewCharacter = Character #characters { currenthp = MaxHP } ,
NewUser = User #egs_user_model { character = NewCharacter , setid = 0 } ,
egs_user_model : write ( NewUser ) ,
%% map change
if User #egs_user_model.areatype =:= mission - >
PrevArea = User #egs_user_model.prev_area ,
event ( { area_change , PrevArea #psu_area.questid , PrevArea #psu_area.zoneid , PrevArea #psu_area.mapid , User #egs_user_model.prev_entryid } , State ) ;
true - > ignore
end ;
%% @todo Forward the mission start to other players of the same party, whatever their location is.
event ( { mission_start , QuestID } , _ State ) - >
log ( " mission start ~b " , [ QuestID ] ) ,
psu_game : send_1020 ( ) ,
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.
event ( { npc_force_invite , NPCid } , #state { gid = GID } ) - >
{ ok , User } = egs_user_model : read ( GID ) ,
%% Create NPC.
log ( " npc force invite ~p " , [ NPCid ] ) ,
TmpNPCUser = psu_npc : user_init ( NPCid , ( ( User #egs_user_model.character ) #characters.mainlevel ) #level.number ) ,
%% Create and join party.
case User #egs_user_model.partypid of
undefined - >
{ ok , PartyPid } = psu_party : start_link ( GID ) ;
PartyPid - >
ignore
end ,
{ ok , PartyPos } = psu_party : join ( PartyPid , npc , TmpNPCUser #egs_user_model.id ) ,
#egs_user_model { instancepid = InstancePid , area = Area , entryid = EntryID , pos = Pos } = User ,
NPCUser = TmpNPCUser #egs_user_model { lid = PartyPos , partypid = PartyPid , instancepid = InstancePid , areatype = mission , area = Area , entryid = EntryID , pos = Pos } ,
egs_user_model : write ( NPCUser ) ,
egs_user_model : write ( User #egs_user_model { partypid = PartyPid } ) ,
%% Send stuff.
Character = NPCUser #egs_user_model.character ,
SentNPCCharacter = Character #characters { gid = NPCid , npcid = NPCid } ,
SentNPCUser = NPCUser #egs_user_model { character = SentNPCCharacter } ,
psu_proto : send_010d ( User , SentNPCUser ) ,
psu_proto : send_0201 ( User , SentNPCUser ) ,
psu_proto : send_0215 ( User , 0 ) ,
psu_game : send_0a04 ( SentNPCUser #egs_user_model.id ) ,
psu_game : send_022c ( 0 , 16#12 ) ,
psu_game : send_1004 ( npc_mission , SentNPCUser , PartyPos ) ,
psu_game : send_100f ( ( SentNPCUser #egs_user_model.character ) #characters.npcid , PartyPos ) ,
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 } ) - >
{ ok , User } = egs_user_model : read ( GID ) ,
%% Create NPC.
log ( " invited npcid ~b " , [ NPCid ] ) ,
TmpNPCUser = psu_npc : user_init ( NPCid , ( ( User #egs_user_model.character ) #characters.mainlevel ) #level.number ) ,
%% Create and join party.
case User #egs_user_model.partypid of
undefined - >
{ ok , PartyPid } = psu_party : start_link ( GID ) ,
psu_game : send_022c ( 0 , 16#12 ) ;
PartyPid - >
ignore
end ,
{ ok , PartyPos } = psu_party : join ( PartyPid , npc , TmpNPCUser #egs_user_model.id ) ,
NPCUser = TmpNPCUser #egs_user_model { lid = PartyPos , partypid = PartyPid } ,
egs_user_model : write ( NPCUser ) ,
egs_user_model : write ( User #egs_user_model { partypid = PartyPid } ) ,
%% Send stuff.
Character = NPCUser #egs_user_model.character ,
SentNPCCharacter = Character #characters { gid = NPCid , npcid = NPCid } ,
SentNPCUser = NPCUser #egs_user_model { character = SentNPCCharacter } ,
psu_game : send_1004 ( npc_invite , SentNPCUser , PartyPos ) ,
psu_game : send_101a ( NPCid , PartyPos ) ;
%% @todo Should be 0115(money) 010a03(confirm sale).
%% @todo We probably need to save the ShopID somewhere since it isn't given back here.
event ( { npc_shop_buy , ShopItemIndex , Quantity } , _ State ) - >
log ( " npc shop buy itemindex ~p quantity ~p " , [ ShopItemIndex , Quantity ] ) ;
%% @todo Currently send the normal items shop for all shops, differentiate.
event ( { npc_shop_enter , ShopID } , #state { gid = GID } ) - >
log ( " npc shop enter ~p " , [ ShopID ] ) ,
case proplists : get_value ( ShopID , ? SHOPS ) of
undefined - > %% @todo Temporary; prevent players from getting stuck.
{ ok , File } = file : read_file ( " p/itemshop.bin " ) ,
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 , File / binary > > ) ;
ItemsList - >
psu_game : send_010a ( ItemsList )
end ;
event ( { npc_shop_leave , ShopID } , #state { gid = GID } ) - >
log ( " npc shop leave ~p " , [ ShopID ] ) ,
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.
event ( { npc_shop_request , ShopID } , _ State ) - >
log ( " npc shop request ~p " , [ ShopID ] ) ,
case ShopID of
80 - > psu_game : send_1a02 ( 0 , 17 , 17 , 3 , 9 ) ; %% lumilass
90 - > psu_game : send_1a02 ( 0 , 5 , 1 , 4 , 5 ) ; %% parum weapon grinding
91 - > psu_game : send_1a02 ( 0 , 5 , 5 , 4 , 7 ) ; %% tenora weapon grinding
92 - > psu_game : send_1a02 ( 0 , 5 , 0 , 4 , 0 ) ; %% yohmei weapon grinding
93 - > psu_game : send_1a02 ( 0 , 5 , 18 , 4 , 0 ) ; %% kubara weapon grinding
_ - > psu_game : send_1a02 ( 0 , 0 , 1 , 0 , 0 )
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 ) ;
%% @todo Second send_1211 argument should be User#egs_user_model.lid. Fix when it's correctly handled.
event ( { object_chair_sit , ObjectTargetID } , _ State ) - >
%~ {ok, User} = egs_user_model:read(get(gid)),
psu_game : send_1211 ( ObjectTargetID , 0 , 8 , 0 ) ;
%% @todo Second psu_game:send_1211 argument should be User#egs_user_model.lid. Fix when it's correctly handled.
event ( { object_chair_stand , ObjectTargetID } , _ State ) - >
%~ {ok, User} = egs_user_model:read(get(gid)),
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 } ) - >
{ ok , User } = egs_user_model : read ( GID ) ,
{ BlockID , EventID } = psu_instance : std_event ( User #egs_user_model.instancepid , ( User #egs_user_model.area ) #psu_area.zoneid , ObjectID ) ,
psu_game : send_1205 ( EventID , BlockID , 0 ) ,
psu_game : send_1213 ( ObjectID , 8 ) ;
event ( { object_key_console_enable , ObjectID } , #state { gid = GID } ) - >
{ ok , User } = egs_user_model : read ( GID ) ,
{ BlockID , [ EventID | _ ] } = psu_instance : std_event ( User #egs_user_model.instancepid , ( User #egs_user_model.area ) #psu_area.zoneid , ObjectID ) ,
psu_game : send_1205 ( EventID , BlockID , 0 ) ,
psu_game : send_1213 ( ObjectID , 1 ) ;
event ( { object_key_console_init , ObjectID } , #state { gid = GID } ) - >
{ ok , User } = egs_user_model : read ( GID ) ,
{ BlockID , [ _ , EventID , _ ] } = psu_instance : std_event ( User #egs_user_model.instancepid , ( User #egs_user_model.area ) #psu_area.zoneid , ObjectID ) ,
psu_game : send_1205 ( EventID , BlockID , 0 ) ;
event ( { object_key_console_open_gate , ObjectID } , #state { gid = GID } ) - >
{ ok , User } = egs_user_model : read ( GID ) ,
{ BlockID , [ _ , _ , EventID ] } = psu_instance : std_event ( User #egs_user_model.instancepid , ( User #egs_user_model.area ) #psu_area.zoneid , ObjectID ) ,
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 } ) - >
{ ok , User } = egs_user_model : read ( GID ) ,
{ BlockID , [ EventID | _ ] } = psu_instance : std_event ( User #egs_user_model.instancepid , ( User #egs_user_model.area ) #psu_area.zoneid , ObjectID ) ,
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 } ) - >
{ ok , User } = egs_user_model : read ( GID ) ,
{ BlockID , EventID } = psu_instance : std_event ( User #egs_user_model.instancepid , ( User #egs_user_model.area ) #psu_area.zoneid , ObjectID ) ,
psu_game : send_1205 ( EventID , BlockID , 1 ) ,
psu_game : send_1213 ( ObjectID , 0 ) ;
event ( { object_switch_on , ObjectID } , #state { gid = GID } ) - >
{ ok , User } = egs_user_model : read ( GID ) ,
{ BlockID , EventID } = psu_instance : std_event ( User #egs_user_model.instancepid , ( User #egs_user_model.area ) #psu_area.zoneid , ObjectID ) ,
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 ) ;
%% @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 } , #state { gid = GID } ) - >
{ ok , User } = egs_user_model : read ( 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 ) ,
psu_game : send_0503 ( User #egs_user_model.pos ) ,
psu_game : send_1211 ( 16#ffffffff , 0 , 14 , 0 ) ;
event ( { party_remove_member , PartyPos } , #state { gid = GID } ) - >
log ( " party remove member ~b " , [ PartyPos ] ) ,
{ ok , DestUser } = egs_user_model : read ( GID ) ,
{ ok , RemovedGID } = psu_party : get_member ( DestUser #egs_user_model.partypid , PartyPos ) ,
psu_party : remove_member ( DestUser #egs_user_model.partypid , PartyPos ) ,
{ ok , RemovedUser } = egs_user_model : read ( RemovedGID ) ,
case ( RemovedUser #egs_user_model.character ) #characters.type of
npc - > egs_user_model : delete ( RemovedGID ) ;
_ - > ignore
end ,
psu_game : send_1006 ( 8 , PartyPos ) ,
psu_game : send_0204 ( DestUser , RemovedUser , 1 ) ,
psu_proto : send_0215 ( DestUser , 0 ) ;
event ( { player_options_change , Options } , #state { gid = GID } ) - >
{ ok , User } = egs_user_model : read ( GID ) ,
file : write_file ( io_lib : format ( " save/ ~s / ~b -character.options " , [ User #egs_user_model.folder , ( User #egs_user_model.character ) #characters.slot ] ) , Options ) ;
%% @todo If the player has a scape, use it! Otherwise red screen.
%% @todo Right now we force revive and don't update the player's HP.
event ( player_death , _ State ) - >
% @todo send_0115(get(gid), 16#ffffffff, LV=1, EXP=idk, Money=1000), % apparently sent everytime you die...
%% use scape:
NewHP = 10 ,
psu_game : send_0117 ( NewHP ) ,
psu_game : send_1022 ( NewHP ) ;
%% red screen with return to lobby choice:
%~ psu_game:send_0111(3, 1);
%% @todo Refill the player's HP to maximum, remove SEs etc.
event ( player_death_return_to_lobby , State = #state { gid = GID } ) - >
{ ok , User } = egs_user_model : read ( GID ) ,
PrevArea = User #egs_user_model.prev_area ,
event ( { area_change , PrevArea #psu_area.questid , PrevArea #psu_area.zoneid , PrevArea #psu_area.mapid , User #egs_user_model.prev_entryid } , State ) ;
event ( player_type_availability_request , _ State ) - >
psu_game : send_1a07 ( ) ;
event ( player_type_capabilities_request , _ State ) - >
psu_game : send_0113 ( ) ;
event ( ppcube_request , _ State ) - >
psu_game : send_1a04 ( ) ;
%% @doc Uni cube handler.
event ( unicube_request , _ State ) - >
psu_game : send_021e ( ) ;
2010-09-19 05:30:31 +08:00
%% @doc Uni selection handler. Selecting anything reloads the character which will then be sent to its associated area.
2010-09-19 01:57:55 +08:00
%% @todo When selecting 'Your room', load a default room.
%% @todo When selecting 'Reload', reload the character in the current lobby.
%% @todo Delete NPC characters and stop the party on entering myroom too.
2010-09-19 05:30:31 +08:00
event ( { unicube_select , Selection , EntryID } , #state { gid = GID } ) - >
2010-09-19 01:57:55 +08:00
case Selection of
cancel - > ignore ;
16#ffffffff - >
log ( " uni selection (my room) " ) ,
psu_game : send_0230 ( ) ,
% 0220
2010-09-19 05:30:31 +08:00
{ ok , User } = egs_user_model : read ( GID ) ,
User2 = User #egs_user_model { area = #psu_area { questid = 1120000 , zoneid = 0 , mapid = 100 } , entryid = 0 } ,
egs_user_model : write ( User2 ) ,
psu_game : char_load ( User2 ) ;
2010-09-19 01:57:55 +08:00
_ UniID - >
log ( " uni selection (reload) " ) ,
psu_game : send_0230 ( ) ,
% 0220
{ ok , User } = egs_user_model : read ( GID ) ,
case User #egs_user_model.partypid of
undefined - >
ignore ;
PartyPid - >
%% @todo Replace stop by leave when leaving stops the party correctly when nobody's there anymore.
%~ psu_party:leave(User#egs_user_model.partypid, User#egs_user_model.id)
{ ok , NPCList } = psu_party : get_npc ( PartyPid ) ,
[ egs_user_model : delete ( NPCGID ) | | { _ Spot , NPCGID } < - NPCList ] ,
psu_party : stop ( PartyPid )
end ,
2010-09-19 05:30:31 +08:00
User2 = User #egs_user_model { entryid = EntryID } ,
egs_user_model : write ( User2 ) ,
psu_game : char_load ( User2 )
2010-09-19 01:57:55 +08:00
end .
%% 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 ) .