Properly handle the LID per player and per zone. Remove related hacks.

This commit is contained in:
Loïc Hoguin 2011-02-19 19:45:50 +01:00
parent 1be3c4f5c6
commit b728731830
7 changed files with 67 additions and 74 deletions

View File

@ -57,7 +57,7 @@
-record(users, { -record(users, {
%% General information. %% General information.
gid :: integer(), gid :: integer(),
lid :: non_neg_integer(), lid = 16#ffff :: 0..16#ffff,
pid :: pid(), pid :: pid(),
time :: integer(), time :: integer(),
character :: tuple(), %% @todo Details. character :: tuple(), %% @todo Details.

View File

@ -268,11 +268,11 @@ event({counter_enter, CounterID, FromZoneID, FromMapID, FromEntryID}, State=#sta
psu_proto:send_0c00(User, State), psu_proto:send_0c00(User, State),
psu_proto:send_020e(QuestData, State), psu_proto:send_020e(QuestData, State),
psu_proto:send_0a05(State), psu_proto:send_0a05(State),
psu_proto:send_010d(User#users{lid=0}, State), psu_proto:send_010d(User, State),
psu_proto:send_0200(0, mission, State), psu_proto:send_0200(0, mission, State),
psu_proto:send_020f(ZoneData, 0, 255, State), psu_proto:send_020f(ZoneData, 0, 255, State),
State2 = State#state{areanb=State#state.areanb + 1}, State2 = State#state{areanb=State#state.areanb + 1},
psu_proto:send_0205(User#users{lid=0}, 0, State2), psu_proto:send_0205(User, 0, State2),
psu_proto:send_100e(CounterID, "Counter", State2), psu_proto:send_100e(CounterID, "Counter", State2),
psu_proto:send_0215(0, State2), psu_proto:send_0215(0, State2),
psu_proto:send_0215(0, State2), psu_proto:send_0215(0, State2),
@ -282,7 +282,7 @@ event({counter_enter, CounterID, FromZoneID, FromMapID, FromEntryID}, State=#sta
psu_game:send_1206(), psu_game:send_1206(),
psu_game:send_1207(), psu_game:send_1207(),
psu_game:send_1212(), psu_game:send_1212(),
psu_proto:send_0201(User#users{lid=0}, State2), psu_proto:send_0201(User, State2),
psu_proto:send_0a06(User, State2), psu_proto:send_0a06(User, State2),
case User#users.partypid of case User#users.partypid of
undefined -> ignore; undefined -> ignore;
@ -353,7 +353,7 @@ event({hit, FromTargetID, ToTargetID, A, B}, State=#state{gid=GID}) ->
end, end,
%% exp %% exp
if HasEXP =:= true -> if HasEXP =:= true ->
psu_proto:send_0115(NewUser#users{lid=0}, ToTargetID, State); psu_proto:send_0115(NewUser, ToTargetID, State);
true -> ignore true -> ignore
end, end,
%% save %% save
@ -547,7 +547,7 @@ event({npc_shop_buy, ShopItemIndex, QuantityOrColor}, State=#state{gid=GID}) ->
egs_users:money_add(GID, -1 * BuyPrice * Quantity), egs_users:money_add(GID, -1 * BuyPrice * Quantity),
ItemUUID = egs_users:item_add(GID, ItemID, Variables), ItemUUID = egs_users:item_add(GID, ItemID, Variables),
{ok, User} = egs_users:read(GID), {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. psu_proto:send_0115(User, State), %% @todo This one is apparently broadcast to everyone in the same zone.
%% @todo Following command isn't done 100% properly. %% @todo Following command isn't done 100% properly.
UCS2Name = << << X:8, 0:8 >> || X <- Name >>, UCS2Name = << << X:8, 0:8 >> || X <- Name >>,
NamePadding = 8 * (46 - byte_size(UCS2Name)), NamePadding = 8 * (46 - byte_size(UCS2Name)),
@ -639,8 +639,8 @@ event({object_healing_pad_tick, [_PartyPos]}, State=#state{gid=GID}) ->
NewHP2 = if NewHP > Character#characters.maxhp -> Character#characters.maxhp; true -> NewHP end, NewHP2 = if NewHP > Character#characters.maxhp -> Character#characters.maxhp; true -> NewHP end,
User2 = User#users{character=Character#characters{currenthp=NewHP2}}, User2 = User#users{character=Character#characters{currenthp=NewHP2}},
egs_users:write(User2), egs_users:write(User2),
psu_proto:send_0117(User2#users{lid=0}, State), psu_proto:send_0117(User2, State),
psu_proto:send_0111(User2#users{lid=0}, 4, State) psu_proto:send_0111(User2, 4, State)
end; end;
event({object_key_console_enable, ObjectID}, #state{gid=GID}) -> event({object_key_console_enable, ObjectID}, #state{gid=GID}) ->
@ -724,7 +724,7 @@ event(player_death, State=#state{gid=GID}) ->
Char = User#users.character, Char = User#users.character,
User2 = User#users{character=Char#characters{currenthp=NewHP}}, User2 = User#users{character=Char#characters{currenthp=NewHP}},
egs_users:write(User2), egs_users:write(User2),
psu_proto:send_0117(User2#users{lid=0}, State), psu_proto:send_0117(User2, State),
psu_proto:send_1022(User2, State); psu_proto:send_1022(User2, State);
%% red screen with return to lobby choice: %% red screen with return to lobby choice:
%~ psu_proto:send_0111(User2, 3, 1, State); %~ psu_proto:send_0111(User2, 3, 1, State);

View File

@ -66,8 +66,7 @@ event(system_game_server_request, State=#state{socket=Socket}) ->
%% @todo Remove the put calls when all the send_xxxx are moved out of psu_game and into psu_proto. %% @todo Remove the put calls when all the send_xxxx are moved out of psu_game and into psu_proto.
event({system_key_auth_request, AuthGID, AuthKey}, State=#state{socket=Socket}) -> event({system_key_auth_request, AuthGID, AuthKey}, State=#state{socket=Socket}) ->
egs_accounts:key_auth(AuthGID, AuthKey), egs_accounts:key_auth(AuthGID, AuthKey),
LID = 1 + mnesia:dirty_update_counter(counters, lobby, 1) rem 1023, egs_users:write(#users{gid=AuthGID, pid=self()}),
egs_users:write(#users{gid=AuthGID, pid=self(), lid=LID}),
put(socket, Socket), put(socket, Socket),
put(gid, AuthGID), put(gid, AuthGID),
State2 = State#state{gid=AuthGID}, State2 = State#state{gid=AuthGID},

View File

@ -18,7 +18,7 @@
%% along with EGS. If not, see <http://www.gnu.org/licenses/>. %% along with EGS. If not, see <http://www.gnu.org/licenses/>.
-module(egs_users). -module(egs_users).
-export([read/1, select/1, write/1, delete/1, item_nth/2, item_add/3, item_qty_add/3, shop_enter/2, shop_leave/1, shop_get/1, money_add/2, broadcast_spawn/2, broadcast_unspawn/2]). -export([read/1, select/1, write/1, delete/1, item_nth/2, item_add/3, item_qty_add/3, shop_enter/2, shop_leave/1, shop_get/1, money_add/2, broadcast_spawn/2, broadcast_unspawn/2, set_zone/3]).
-define(TABLE, users). -define(TABLE, users).
@ -177,3 +177,7 @@ broadcast_unspawn(GID, PlayersGID) ->
{ok, DestUser} = read(DestGID), {ok, DestUser} = read(DestGID),
DestUser#users.pid ! {egs, player_unspawn, OrigUser} DestUser#users.pid ! {egs, player_unspawn, OrigUser}
end, PlayersGID). end, PlayersGID).
set_zone(GID, ZonePid, LID) ->
{ok, User} = read(GID),
write(User#users{zonepid=ZonePid, lid=LID}).

View File

@ -68,9 +68,10 @@ handle_call(setid, _From, State) ->
{reply, State#state.setid, State}; {reply, State#state.setid, State};
handle_call({enter, GID}, _From, State) -> handle_call({enter, GID}, _From, State) ->
[LID|FreeLIDs] = State#state.freelids,
egs_users:set_zone(GID, self(), LID),
Players = State#state.players, Players = State#state.players,
PlayersGID = players_gid(Players), PlayersGID = players_gid(Players),
[LID|FreeLIDs] = State#state.freelids,
egs_users:broadcast_spawn(GID, PlayersGID), egs_users:broadcast_spawn(GID, PlayersGID),
{reply, LID, State#state{players=[{GID, LID}|Players], freelids=FreeLIDs}}; {reply, LID, State#state{players=[{GID, LID}|Players], freelids=FreeLIDs}};

View File

@ -59,23 +59,27 @@ area_load(QuestID, ZoneID, MapID, EntryID, State) ->
true -> User true -> User
end, end,
%% Load the zone. %% Load the zone.
User3 = if ZoneChange -> State1 = if ZoneChange ->
ZonePid = egs_quests:zone_pid(User2#users.questpid, ZoneID), ZonePid = egs_quests:zone_pid(User2#users.questpid, ZoneID),
egs_zones:leave(User2#users.zonepid, User2#users.gid), egs_zones:leave(User2#users.zonepid, User2#users.gid),
LID = egs_zones:enter(ZonePid, User2#users.gid), NewLID = egs_zones:enter(ZonePid, User2#users.gid),
psu_proto:send_0a05(State), NewState = State#state{lid=NewLID},
psu_proto:send_0111(User2#users{lid=0}, 6, State), {ok, User3} = egs_users:read(User2#users.gid),
psu_proto:send_010d(User2#users{lid=0}, State), psu_proto:send_0a05(NewState),
psu_proto:send_0200(ZoneID, AreaType, State), psu_proto:send_0111(User3, 6, NewState),
psu_proto:send_020f(egs_quests_db:zone_nbl(QuestID, ZoneID), egs_zones:setid(ZonePid), SeasonID, State), psu_proto:send_010d(User3, NewState),
User2#users{zonepid=ZonePid}; psu_proto:send_0200(ZoneID, AreaType, NewState),
true -> User2 psu_proto:send_020f(egs_quests_db:zone_nbl(QuestID, ZoneID), egs_zones:setid(ZonePid), SeasonID, NewState),
NewState;
true ->
User3 = User2,
State
end, end,
%% Save the user. %% Save the user.
egs_users:write(User3), egs_users:write(User3),
%% Load the player location. %% Load the player location.
State2 = State#state{areanb=State#state.areanb + 1}, State2 = State1#state{areanb=State#state.areanb + 1},
psu_proto:send_0205(User3#users{lid=0}, IsSeasonal, State2), psu_proto:send_0205(User3, IsSeasonal, State2),
psu_proto:send_100e(User3#users.area, User3#users.entryid, AreaShortName, State2), psu_proto:send_100e(User3#users.area, User3#users.entryid, AreaShortName, State2),
%% Load the zone objects. %% Load the zone objects.
if ZoneChange -> if ZoneChange ->
@ -83,7 +87,7 @@ area_load(QuestID, ZoneID, MapID, EntryID, State) ->
true -> ignore true -> ignore
end, end,
%% Load the player. %% Load the player.
psu_proto:send_0201(User3#users{lid=0}, State2), psu_proto:send_0201(User3, State2),
if ZoneChange -> if ZoneChange ->
psu_proto:send_0a06(User3, State2), psu_proto:send_0a06(User3, State2),
%% Load the other players in the zone. %% Load the other players in the zone.

View File

@ -1196,12 +1196,11 @@ parse_hits(Hits, Acc) ->
%% @doc Send character appearance and other information. %% @doc Send character appearance and other information.
%% @todo Probably don't pattern match the data like this... %% @todo Probably don't pattern match the data like this...
%% @todo Handle the DestLID properly. send_010d(CharUser, #state{socket=Socket, gid=DestGID, lid=DestLID}) ->
send_010d(CharUser, #state{socket=Socket, gid=DestGID}) ->
CharGID = CharUser#users.gid, CharGID = CharUser#users.gid,
CharLID = CharUser#users.lid, CharLID = CharUser#users.lid,
<< _:640, CharBin/bits >> = psu_characters:character_user_to_binary(CharUser), << _:640, CharBin/bits >> = psu_characters:character_user_to_binary(CharUser),
packet_send(Socket, << 16#010d0300:32, 0:160, 16#00011300:32, DestGID:32/little, packet_send(Socket, << 16#010d0300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little,
0:64, 1:32/little, 0:32, 16#00000300:32, 16#ffff0000:32, 0:32, CharGID:32/little, 0:64, 1:32/little, 0:32, 16#00000300:32, 16#ffff0000:32, 0:32, CharGID:32/little,
0:192, CharGID:32/little, CharLID:32/little, 16#ffffffff:32, CharBin/binary >>). 0:192, CharGID:32/little, CharLID:32/little, 16#ffffffff:32, CharBin/binary >>).
@ -1241,20 +1240,18 @@ send_0117(#users{gid=CharGID, lid=CharLID, character=#characters{currenthp=HP}},
CharGID:32/little, CharLID:32/little, SE/binary, HP:32/little, 0:32 >>). CharGID:32/little, CharLID:32/little, SE/binary, HP:32/little, 0:32 >>).
%% @doc Send the zone initialization command. %% @doc Send the zone initialization command.
%% @todo Handle the LID properly in both places.
%% @todo Handle NbPlayers properly. There's more than 1 player! %% @todo Handle NbPlayers properly. There's more than 1 player!
send_0200(ZoneID, ZoneType, #state{socket=Socket, gid=DestGID}) -> send_0200(ZoneID, ZoneType, #state{socket=Socket, gid=DestGID, lid=DestLID}) ->
Var = case ZoneType of Var = case ZoneType of
mission -> << 16#06000500:32, 16#01000000:32, 0:64, 16#00040000:32, 16#00010000:32, 16#00140000:32 >>; mission -> << 16#06000500:32, 16#01000000:32, 0:64, 16#00040000:32, 16#00010000:32, 16#00140000:32 >>;
myroom -> << 16#06000000:32, 16#02000000:32, 0:64, 16#40000000:32, 16#00010000:32, 16#00010000:32 >>; myroom -> << 16#06000000:32, 16#02000000:32, 0:64, 16#40000000:32, 16#00010000:32, 16#00010000:32 >>;
_ -> << 16#00040000:32, 0:160, 16#00140000:32 >> _ -> << 16#00040000:32, 0:160, 16#00140000:32 >>
end, end,
packet_send(Socket, << 16#02000300:32, 0:160, 16#00011300:32, DestGID:32/little, 0:64, packet_send(Socket, << 16#02000300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:64,
0:16, ZoneID:16/little, 1:32/little, 16#ffffffff:32, Var/binary, 16#ffffffff:32, 16#ffffffff:32 >>). DestLID:16/little, ZoneID:16/little, 1:32/little, 16#ffffffff:32, Var/binary, 16#ffffffff:32, 16#ffffffff:32 >>).
%% @doc Send character location, appearance and other information. %% @doc Send character location, appearance and other information.
%% @todo Handle the DestLID properly. send_0201(CharUser, #state{socket=Socket, gid=DestGID, lid=DestLID}) ->
send_0201(CharUser, #state{socket=Socket, gid=DestGID}) ->
[CharTypeID, GameVersion] = case (CharUser#users.character)#characters.type of [CharTypeID, GameVersion] = case (CharUser#users.character)#characters.type of
npc -> [16#00001d00, 255]; npc -> [16#00001d00, 255];
_ -> [16#00001200, 0] _ -> [16#00001200, 0]
@ -1263,7 +1260,7 @@ send_0201(CharUser, #state{socket=Socket, gid=DestGID}) ->
CharBin = psu_characters:character_user_to_binary(CharUser), CharBin = psu_characters:character_user_to_binary(CharUser),
IsGM = 0, IsGM = 0,
OnlineStatus = 0, OnlineStatus = 0,
packet_send(Socket, << 16#02010300:32, 0:32, CharTypeID:32, CharGID:32/little, packet_send(Socket, << 16#02010300:32, DestLID:16/little, 0:16, CharTypeID:32, CharGID:32/little,
0:64, 16#00011300:32, DestGID:32/little, 0:64, CharBin/binary, IsGM:8, 0:8, OnlineStatus:8, GameVersion:8, 0:608 >>). 0:64, 16#00011300:32, DestGID:32/little, 0:64, CharBin/binary, IsGM:8, 0:8, OnlineStatus:8, GameVersion:8, 0:608 >>).
%% @doc Hello command. Sent when a client connects to the game or login server. %% @doc Hello command. Sent when a client connects to the game or login server.
@ -1272,21 +1269,19 @@ send_0202(#state{socket=Socket, gid=DestGID, lid=DestLID}) ->
packet_send(Socket, << 16#020203bf:32, DestLID:16/little, 0:272, DestGID:32/little, 0:1024 >>). packet_send(Socket, << 16#020203bf:32, DestLID:16/little, 0:272, DestGID:32/little, 0:1024 >>).
%% @doc Spawn a player with the given GID and LID. %% @doc Spawn a player with the given GID and LID.
%% @todo Handle the LID properly. send_0203(#users{gid=CharGID, lid=CharLID}, #state{socket=Socket, gid=DestGID, lid=DestLID}) ->
send_0203(#users{gid=CharGID, lid=CharLID}, #state{socket=Socket, gid=DestGID}) -> packet_send(Socket, << 16#02030300:32, DestLID:16/little, 0:144, 16#00011300:32,
packet_send(Socket, << 16#02030300:32, 0:160, 16#00011300:32,
DestGID:32/little, 0:64, CharGID:32/little, CharLID:32/little >>). DestGID:32/little, 0:64, CharGID:32/little, CharLID:32/little >>).
%% @doc Unspawn the given character. %% @doc Unspawn the given character.
%% @todo LID.
%% @todo The last 4 bytes are probably the number of players remaining in the zone. %% @todo The last 4 bytes are probably the number of players remaining in the zone.
send_0204(User, #state{socket=Socket, gid=DestGID}) -> send_0204(User, #state{socket=Socket, gid=DestGID, lid=DestLID}) ->
CharTypeID = case (User#users.character)#characters.type of CharTypeID = case (User#users.character)#characters.type of
npc -> 16#00001d00; npc -> 16#00001d00;
_ -> 16#00001200 _ -> 16#00001200
end, end,
#users{gid=CharGID, lid=CharLID} = User, #users{gid=CharGID, lid=CharLID} = User,
packet_send(Socket, << 16#02040300:32, 0:32, CharTypeID:32, CharGID:32/little, 0:64, packet_send(Socket, << 16#02040300:32, DestLID:16/little, 0:16, CharTypeID:32, CharGID:32/little, 0:64,
16#00011300:32, DestGID:32/little, 0:64, CharGID:32/little, CharLID:32/little, 100:32/little >>). 16#00011300:32, DestGID:32/little, 0:64, CharGID:32/little, CharLID:32/little, 100:32/little >>).
%% @doc Make the client load a new map. %% @doc Make the client load a new map.
@ -1296,9 +1291,8 @@ send_0205(CharUser, IsSeasonal, #state{socket=Socket, gid=DestGID, lid=DestLID,
16#ffffffff:32, ZoneID:32/little, MapID:32/little, EntryID:32/little, AreaNb:32/little, CharLID:16/little, 0:8, IsSeasonal:8 >>). 16#ffffffff:32, ZoneID:32/little, MapID:32/little, EntryID:32/little, AreaNb:32/little, CharLID:16/little, 0:8, IsSeasonal:8 >>).
%% @doc Indicate to the client that loading should finish. %% @doc Indicate to the client that loading should finish.
%% @todo Handle the DestLID properly. send_0208(#state{socket=Socket, gid=DestGID, lid=DestLID, areanb=AreaNb}) ->
send_0208(#state{socket=Socket, gid=DestGID, areanb=AreaNb}) -> packet_send(Socket, << 16#02080300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:64, AreaNb:32/little >>).
packet_send(Socket, << 16#02080300:32, 0:160, 16#00011300:32, DestGID:32/little, 0:64, AreaNb:32/little >>).
%% @todo No idea what this one does. For unknown reasons it uses channel 2. %% @todo No idea what this one does. For unknown reasons it uses channel 2.
%% @todo Handle the DestLID properly? %% @todo Handle the DestLID properly?
@ -1400,11 +1394,11 @@ send_0231(URL, #state{socket=Socket, gid=DestGID, lid=DestLID}) ->
packet_send(Socket, << 16#02310300:32, DestLID:16/little, 0:16, 16#00000f00:32, DestGID:32/little, 0:64, packet_send(Socket, << 16#02310300:32, DestLID:16/little, 0:16, 16#00000f00:32, DestGID:32/little, 0:64,
16#00000f00:32, DestGID:32/little, 0:64, Length:32/little, URLBin/binary, 0:Padding >>). 16#00000f00:32, DestGID:32/little, 0:64, Length:32/little, URLBin/binary, 0:Padding >>).
%% @todo Handle the LID properly. %% @doc Send the list of players already spawned in the zone when entering it.
send_0233(Users, #state{socket=Socket, gid=DestGID}) -> send_0233(Users, #state{socket=Socket, gid=DestGID, lid=DestLID}) ->
NbUsers = length(Users), NbUsers = length(Users),
Bin = build_0233_users(Users, []), Bin = build_0233_users(Users, []),
packet_send(Socket, << 16#02330300:32, 0:32, 16#00001200:32, DestGID:32/little, 0:64, packet_send(Socket, << 16#02330300:32, DestLID:16/little, 0:16, 16#00001200:32, DestGID:32/little, 0:64,
16#00011300:32, DestGID:32/little, 0:64, NbUsers:32/little, Bin/binary, 0:608 >>). 16#00011300:32, DestGID:32/little, 0:64, NbUsers:32/little, Bin/binary, 0:608 >>).
build_0233_users([], Acc) -> build_0233_users([], Acc) ->
@ -1414,32 +1408,28 @@ build_0233_users([User|Tail], Acc) ->
build_0233_users(Tail, [<< Bin/binary, 0:32 >>|Acc]). build_0233_users(Tail, [<< Bin/binary, 0:32 >>|Acc]).
%% @doc Start the zone handling: load the zone file and the objects sent separately. %% @doc Start the zone handling: load the zone file and the objects sent separately.
%% @todo Handle the LID properly. send_0236(#state{socket=Socket, gid=DestGID, lid=DestLID}) ->
send_0236(#state{socket=Socket, gid=DestGID}) -> packet_send(Socket, << 16#02360300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:64 >>).
packet_send(Socket, << 16#02360300:32, 0:160, 16#00011300:32, DestGID:32/little, 0:64 >>).
%% @doc Chat message. %% @doc Chat message.
%% @todo Handle the LID properly. send_0304(FromGID, ChatTypeID, ChatGID, ChatName, ChatModifiers, ChatMessage, #state{socket=Socket, gid=DestGID, lid=DestLID}) ->
send_0304(FromGID, ChatTypeID, ChatGID, ChatName, ChatModifiers, ChatMessage, #state{socket=Socket, gid=DestGID}) ->
{chat_modifiers, ChatType, ChatCutIn, ChatCutInAngle, ChatMsgLength, ChatChannel, ChatCharacterType} = ChatModifiers, {chat_modifiers, ChatType, ChatCutIn, ChatCutInAngle, ChatMsgLength, ChatChannel, ChatCharacterType} = ChatModifiers,
packet_send(Socket, << 16#03040300:32, 0:32, 16#00011300:32, FromGID:32/little, 0:64, 16#00011300:32, DestGID:32/little, 0:64, packet_send(Socket, << 16#03040300:32, DestLID:16/little, 0:16, 16#00011300:32, FromGID:32/little, 0:64, 16#00011300:32, DestGID:32/little, 0:64,
ChatTypeID:32, ChatGID:32/little, 0:64, ChatType:8, ChatCutIn:8, ChatCutInAngle:8, ChatMsgLength:8, ChatTypeID:32, ChatGID:32/little, 0:64, ChatType:8, ChatCutIn:8, ChatCutInAngle:8, ChatMsgLength:8,
ChatChannel:8, ChatCharacterType:8, 0:16, ChatName/binary, ChatMessage/binary >>). ChatChannel:8, ChatCharacterType:8, 0:16, ChatName/binary, ChatMessage/binary >>).
%% @todo Inventory related. Doesn't seem to do anything. %% @todo Inventory related. Doesn't seem to do anything.
%% @todo Handle the LID properly. send_0a05(#state{socket=Socket, gid=DestGID, lid=DestLID}) ->
send_0a05(#state{socket=Socket, gid=DestGID}) -> packet_send(Socket, << 16#0a050300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:64 >>).
packet_send(Socket, << 16#0a050300:32, 0:160, 16#00011300:32, DestGID:32/little, 0:64 >>).
%% @doc Send the list of ItemUUID for the items in the inventory. %% @doc Send the list of ItemUUID for the items in the inventory.
%% @todo Handle the LID properly. send_0a06(CharUser, #state{socket=Socket, gid=DestGID, lid=DestLID}) ->
send_0a06(CharUser, #state{socket=Socket, gid=DestGID}) ->
Len = length((CharUser#users.character)#characters.inventory), Len = length((CharUser#users.character)#characters.inventory),
UUIDs = lists:seq(1, Len), UUIDs = lists:seq(1, Len),
Bin = iolist_to_binary([ << N:32/little >> || N <- UUIDs]), Bin = iolist_to_binary([ << N:32/little >> || N <- UUIDs]),
Blanks = lists:seq(1, 60 - Len), Blanks = lists:seq(1, 60 - Len),
Bin2 = iolist_to_binary([ << 16#ffffffff:32 >> || _N <- Blanks]), Bin2 = iolist_to_binary([ << 16#ffffffff:32 >> || _N <- Blanks]),
packet_send(Socket, << 16#0a060300:32, 0:64, DestGID:32/little, 0:64, 16#00011300:32, DestGID:32/little, 0:64, Bin/binary, Bin2/binary >>). packet_send(Socket, << 16#0a060300:32, DestLID:16/little, 0:48, DestGID:32/little, 0:64, 16#00011300:32, DestGID:32/little, 0:64, Bin/binary, Bin2/binary >>).
%% @doc Send an item's description. %% @doc Send an item's description.
send_0a11(ItemID, ItemDesc, #state{socket=Socket, gid=DestGID, lid=DestLID}) -> send_0a11(ItemID, ItemDesc, #state{socket=Socket, gid=DestGID, lid=DestLID}) ->
@ -1466,10 +1456,9 @@ send_0c08(#state{socket=Socket, gid=DestGID}) ->
packet_send(Socket, << 16#0c080300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:96 >>). packet_send(Socket, << 16#0c080300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:96 >>).
%% @doc Send the counter's mission options (0 = invisible, 2 = disabled, 3 = available). %% @doc Send the counter's mission options (0 = invisible, 2 = disabled, 3 = available).
%% @todo LID. send_0c10(Options, #state{socket=Socket, gid=DestGID, lid=DestLID}) ->
send_0c10(Options, #state{socket=Socket, gid=DestGID}) ->
Size = byte_size(Options), Size = byte_size(Options),
packet_send(Socket, << 16#0c100300:32, 0:160, 16#00011300:32, DestGID:32/little, 0:64, 1, 0, Size:16/little, Options/binary >>). packet_send(Socket, << 16#0c100300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:64, 1, 0, Size:16/little, Options/binary >>).
%% @doc Send the general data and flags for the selected character. %% @doc Send the general data and flags for the selected character.
%% @todo Handle bitflags and value flags properly. %% @todo Handle bitflags and value flags properly.
@ -1543,9 +1532,8 @@ send_1022(#users{character=#characters{currenthp=HP}}, #state{socket=Socket, gid
packet_send(Socket, << 16#10220300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:64, HP:32/little, PartyPos:32/little >>). packet_send(Socket, << 16#10220300:32, 16#ffff:16, 0:144, 16#00011300:32, DestGID:32/little, 0:64, HP:32/little, PartyPos:32/little >>).
%% @todo Always the same value, no idea what it's for. %% @todo Always the same value, no idea what it's for.
%% @todo Handle the LID properly. send_1204(#state{socket=Socket, gid=DestGID, lid=DestLID}) ->
send_1204(#state{socket=Socket, gid=DestGID}) -> packet_send(Socket, << 16#12040300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:96, 16#20000000:32, 0:256 >>).
packet_send(Socket, << 16#12040300:32, 0:160, 16#00011300:32, DestGID:32/little, 0:96, 16#20000000:32, 0:256 >>).
%% @doc Send the player's partner card. %% @doc Send the player's partner card.
%% @todo Handle the LID and comment properly. %% @todo Handle the LID and comment properly.
@ -1570,26 +1558,23 @@ send_1701(#state{socket=Socket, gid=DestGID}) ->
packet_send(Socket, << 16#17010300:32, 0:160, 16#00011300:32, DestGID:32/little, 0:96 >>). packet_send(Socket, << 16#17010300:32, 0:160, 16#00011300:32, DestGID:32/little, 0:96 >>).
%% @doc Send the background to use for the counter. %% @doc Send the background to use for the counter.
%% @todo Handle LID properly. send_1711(Bg, #state{socket=Socket, gid=DestGID, lid=DestLID}) ->
send_1711(Bg, #state{socket=Socket, gid=DestGID}) -> packet_send(Socket, << 16#17110300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:64, Bg:8, 0:24 >>).
packet_send(Socket, << 16#17110300:32, 0:160, 16#00011300:32, DestGID:32/little, 0:64, Bg:8, 0:24 >>).
%% @doc NPC shop request reply. %% @doc NPC shop request reply.
%% @todo Handle the LID properly. send_1a02(A, B, C, D, #state{socket=Socket, gid=DestGID, lid=DestLID}) ->
send_1a02(A, B, C, D, #state{socket=Socket, gid=DestGID}) -> packet_send(Socket, << 16#1a020300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:96,
packet_send(Socket, << 16#1a020300:32, 0:160, 16#00011300:32, DestGID:32/little, 0:96,
A:16/little, B:16/little, C:16/little, D:16/little >>). A:16/little, B:16/little, C:16/little, D:16/little >>).
%% @doc Lumilass available hairstyles/headtypes handler. %% @doc Lumilass available hairstyles/headtypes handler.
%% @todo Handle the LID properly. send_1a03(CharUser, #state{socket=Socket, gid=DestGID, lid=DestLID}) ->
send_1a03(CharUser, #state{socket=Socket, gid=DestGID}) ->
{ok, Conf} = file:consult("priv/lumilass.conf"), {ok, Conf} = file:consult("priv/lumilass.conf"),
Character = CharUser#users.character, Character = CharUser#users.character,
NbHeadtypes = proplists:get_value({headtypes, Character#characters.gender, Character#characters.race}, Conf, 0), NbHeadtypes = proplists:get_value({headtypes, Character#characters.gender, Character#characters.race}, Conf, 0),
HairstylesList = proplists:get_value({hairstyles, Character#characters.gender}, Conf), HairstylesList = proplists:get_value({hairstyles, Character#characters.gender}, Conf),
NbHairstyles = length(HairstylesList), NbHairstyles = length(HairstylesList),
HairstylesBin = iolist_to_binary([ << N:32 >> || N <- HairstylesList]), HairstylesBin = iolist_to_binary([ << N:32 >> || N <- HairstylesList]),
packet_send(Socket, << 16#1a030300:32, 0:160, 16#00011300:32, DestGID:32/little, 0:96, packet_send(Socket, << 16#1a030300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:96,
NbHairstyles:32/little, NbHeadtypes:32/little, 0:416, HairstylesBin/binary, 0:32 >>). NbHairstyles:32/little, NbHeadtypes:32/little, 0:416, HairstylesBin/binary, 0:32 >>).
%% @doc Available types handler. Enable all 16 types. %% @doc Available types handler. Enable all 16 types.