From b728731830b74dec648ab781daeaa8d8a4380d81 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Lo=C3=AFc=20Hoguin?= Date: Sat, 19 Feb 2011 19:45:50 +0100 Subject: [PATCH] Properly handle the LID per player and per zone. Remove related hacks. --- include/records.hrl | 2 +- src/egs_game.erl | 16 ++++----- src/egs_login.erl | 3 +- src/egs_users.erl | 6 +++- src/egs_zones.erl | 3 +- src/psu/psu_game.erl | 28 ++++++++------- src/psu/psu_proto.erl | 83 ++++++++++++++++++------------------------- 7 files changed, 67 insertions(+), 74 deletions(-) diff --git a/include/records.hrl b/include/records.hrl index 7760025..8730a13 100644 --- a/include/records.hrl +++ b/include/records.hrl @@ -57,7 +57,7 @@ -record(users, { %% General information. gid :: integer(), - lid :: non_neg_integer(), + lid = 16#ffff :: 0..16#ffff, pid :: pid(), time :: integer(), character :: tuple(), %% @todo Details. diff --git a/src/egs_game.erl b/src/egs_game.erl index c138d7c..c4be3f1 100644 --- a/src/egs_game.erl +++ b/src/egs_game.erl @@ -268,11 +268,11 @@ event({counter_enter, CounterID, FromZoneID, FromMapID, FromEntryID}, State=#sta psu_proto:send_0c00(User, State), psu_proto:send_020e(QuestData, 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_020f(ZoneData, 0, 255, State), 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_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_1207(), 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), case User#users.partypid of undefined -> ignore; @@ -353,7 +353,7 @@ event({hit, FromTargetID, ToTargetID, A, B}, State=#state{gid=GID}) -> end, %% exp if HasEXP =:= true -> - psu_proto:send_0115(NewUser#users{lid=0}, ToTargetID, State); + psu_proto:send_0115(NewUser, ToTargetID, State); true -> ignore end, %% save @@ -547,7 +547,7 @@ event({npc_shop_buy, ShopItemIndex, QuantityOrColor}, State=#state{gid=GID}) -> 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. + 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. UCS2Name = << << X:8, 0:8 >> || X <- Name >>, 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, 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) + psu_proto:send_0117(User2, State), + psu_proto:send_0111(User2, 4, State) end; event({object_key_console_enable, ObjectID}, #state{gid=GID}) -> @@ -724,7 +724,7 @@ event(player_death, State=#state{gid=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), + psu_proto:send_0117(User2, State), psu_proto:send_1022(User2, State); %% red screen with return to lobby choice: %~ psu_proto:send_0111(User2, 3, 1, State); diff --git a/src/egs_login.erl b/src/egs_login.erl index 037eb20..d6855ca 100644 --- a/src/egs_login.erl +++ b/src/egs_login.erl @@ -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. event({system_key_auth_request, AuthGID, AuthKey}, State=#state{socket=Socket}) -> 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(), lid=LID}), + egs_users:write(#users{gid=AuthGID, pid=self()}), put(socket, Socket), put(gid, AuthGID), State2 = State#state{gid=AuthGID}, diff --git a/src/egs_users.erl b/src/egs_users.erl index 23362a8..b0a4bb3 100644 --- a/src/egs_users.erl +++ b/src/egs_users.erl @@ -18,7 +18,7 @@ %% along with EGS. If not, see . -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). @@ -177,3 +177,7 @@ broadcast_unspawn(GID, PlayersGID) -> {ok, DestUser} = read(DestGID), DestUser#users.pid ! {egs, player_unspawn, OrigUser} end, PlayersGID). + +set_zone(GID, ZonePid, LID) -> + {ok, User} = read(GID), + write(User#users{zonepid=ZonePid, lid=LID}). diff --git a/src/egs_zones.erl b/src/egs_zones.erl index e318db5..93ccccc 100644 --- a/src/egs_zones.erl +++ b/src/egs_zones.erl @@ -68,9 +68,10 @@ handle_call(setid, _From, State) -> {reply, State#state.setid, State}; handle_call({enter, GID}, _From, State) -> + [LID|FreeLIDs] = State#state.freelids, + egs_users:set_zone(GID, self(), LID), Players = State#state.players, PlayersGID = players_gid(Players), - [LID|FreeLIDs] = State#state.freelids, egs_users:broadcast_spawn(GID, PlayersGID), {reply, LID, State#state{players=[{GID, LID}|Players], freelids=FreeLIDs}}; diff --git a/src/psu/psu_game.erl b/src/psu/psu_game.erl index 21e8de4..4c38bbd 100644 --- a/src/psu/psu_game.erl +++ b/src/psu/psu_game.erl @@ -59,23 +59,27 @@ area_load(QuestID, ZoneID, MapID, EntryID, State) -> true -> User end, %% Load the zone. - User3 = if ZoneChange -> + State1 = if ZoneChange -> ZonePid = egs_quests:zone_pid(User2#users.questpid, ZoneID), egs_zones:leave(User2#users.zonepid, User2#users.gid), - LID = egs_zones:enter(ZonePid, User2#users.gid), - psu_proto:send_0a05(State), - psu_proto:send_0111(User2#users{lid=0}, 6, State), - psu_proto:send_010d(User2#users{lid=0}, State), - psu_proto:send_0200(ZoneID, AreaType, State), - psu_proto:send_020f(egs_quests_db:zone_nbl(QuestID, ZoneID), egs_zones:setid(ZonePid), SeasonID, State), - User2#users{zonepid=ZonePid}; - true -> User2 + NewLID = egs_zones:enter(ZonePid, User2#users.gid), + NewState = State#state{lid=NewLID}, + {ok, User3} = egs_users:read(User2#users.gid), + psu_proto:send_0a05(NewState), + psu_proto:send_0111(User3, 6, NewState), + psu_proto:send_010d(User3, NewState), + psu_proto:send_0200(ZoneID, AreaType, NewState), + psu_proto:send_020f(egs_quests_db:zone_nbl(QuestID, ZoneID), egs_zones:setid(ZonePid), SeasonID, NewState), + NewState; + true -> + User3 = User2, + State end, %% Save the user. egs_users:write(User3), %% Load the player location. - State2 = State#state{areanb=State#state.areanb + 1}, - psu_proto:send_0205(User3#users{lid=0}, IsSeasonal, State2), + State2 = State1#state{areanb=State#state.areanb + 1}, + psu_proto:send_0205(User3, IsSeasonal, State2), psu_proto:send_100e(User3#users.area, User3#users.entryid, AreaShortName, State2), %% Load the zone objects. if ZoneChange -> @@ -83,7 +87,7 @@ area_load(QuestID, ZoneID, MapID, EntryID, State) -> true -> ignore end, %% Load the player. - psu_proto:send_0201(User3#users{lid=0}, State2), + psu_proto:send_0201(User3, State2), if ZoneChange -> psu_proto:send_0a06(User3, State2), %% Load the other players in the zone. diff --git a/src/psu/psu_proto.erl b/src/psu/psu_proto.erl index 8a082ad..87d72ca 100644 --- a/src/psu/psu_proto.erl +++ b/src/psu/psu_proto.erl @@ -1196,12 +1196,11 @@ parse_hits(Hits, Acc) -> %% @doc Send character appearance and other information. %% @todo Probably don't pattern match the data like this... -%% @todo Handle the DestLID properly. -send_010d(CharUser, #state{socket=Socket, gid=DestGID}) -> +send_010d(CharUser, #state{socket=Socket, gid=DestGID, lid=DestLID}) -> CharGID = CharUser#users.gid, CharLID = CharUser#users.lid, << _: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: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 >>). %% @doc Send the zone initialization command. -%% @todo Handle the LID properly in both places. %% @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 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 >>; _ -> << 16#00040000:32, 0:160, 16#00140000:32 >> end, - packet_send(Socket, << 16#02000300:32, 0:160, 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 >>). + packet_send(Socket, << 16#02000300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:64, + 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. -%% @todo Handle the DestLID properly. -send_0201(CharUser, #state{socket=Socket, gid=DestGID}) -> +send_0201(CharUser, #state{socket=Socket, gid=DestGID, lid=DestLID}) -> [CharTypeID, GameVersion] = case (CharUser#users.character)#characters.type of npc -> [16#00001d00, 255]; _ -> [16#00001200, 0] @@ -1263,7 +1260,7 @@ send_0201(CharUser, #state{socket=Socket, gid=DestGID}) -> CharBin = psu_characters:character_user_to_binary(CharUser), IsGM = 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 >>). %% @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 >>). %% @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}) -> - packet_send(Socket, << 16#02030300:32, 0:160, 16#00011300:32, +send_0203(#users{gid=CharGID, lid=CharLID}, #state{socket=Socket, gid=DestGID, lid=DestLID}) -> + packet_send(Socket, << 16#02030300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:64, CharGID:32/little, CharLID:32/little >>). %% @doc Unspawn the given character. -%% @todo LID. %% @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 npc -> 16#00001d00; _ -> 16#00001200 end, #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 >>). %% @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 >>). %% @doc Indicate to the client that loading should finish. -%% @todo Handle the DestLID properly. -send_0208(#state{socket=Socket, gid=DestGID, areanb=AreaNb}) -> - packet_send(Socket, << 16#02080300:32, 0:160, 16#00011300:32, DestGID:32/little, 0:64, AreaNb:32/little >>). +send_0208(#state{socket=Socket, gid=DestGID, lid=DestLID, areanb=AreaNb}) -> + packet_send(Socket, << 16#02080300:32, DestLID:16/little, 0:144, 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 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, 16#00000f00:32, DestGID:32/little, 0:64, Length:32/little, URLBin/binary, 0:Padding >>). -%% @todo Handle the LID properly. -send_0233(Users, #state{socket=Socket, gid=DestGID}) -> +%% @doc Send the list of players already spawned in the zone when entering it. +send_0233(Users, #state{socket=Socket, gid=DestGID, lid=DestLID}) -> NbUsers = length(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 >>). build_0233_users([], Acc) -> @@ -1414,32 +1408,28 @@ build_0233_users([User|Tail], Acc) -> build_0233_users(Tail, [<< Bin/binary, 0:32 >>|Acc]). %% @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}) -> - packet_send(Socket, << 16#02360300:32, 0:160, 16#00011300:32, DestGID:32/little, 0:64 >>). +send_0236(#state{socket=Socket, gid=DestGID, lid=DestLID}) -> + packet_send(Socket, << 16#02360300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:64 >>). %% @doc Chat message. -%% @todo Handle the LID properly. -send_0304(FromGID, ChatTypeID, ChatGID, ChatName, ChatModifiers, ChatMessage, #state{socket=Socket, gid=DestGID}) -> +send_0304(FromGID, ChatTypeID, ChatGID, ChatName, ChatModifiers, ChatMessage, #state{socket=Socket, gid=DestGID, lid=DestLID}) -> {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, ChatChannel:8, ChatCharacterType:8, 0:16, ChatName/binary, ChatMessage/binary >>). %% @todo Inventory related. Doesn't seem to do anything. -%% @todo Handle the LID properly. -send_0a05(#state{socket=Socket, gid=DestGID}) -> - packet_send(Socket, << 16#0a050300:32, 0:160, 16#00011300:32, DestGID:32/little, 0:64 >>). +send_0a05(#state{socket=Socket, gid=DestGID, lid=DestLID}) -> + packet_send(Socket, << 16#0a050300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:64 >>). %% @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}) -> +send_0a06(CharUser, #state{socket=Socket, gid=DestGID, lid=DestLID}) -> Len = length((CharUser#users.character)#characters.inventory), UUIDs = lists:seq(1, Len), Bin = iolist_to_binary([ << N:32/little >> || N <- UUIDs]), Blanks = lists:seq(1, 60 - Len), 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. 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 >>). %% @doc Send the counter's mission options (0 = invisible, 2 = disabled, 3 = available). -%% @todo LID. -send_0c10(Options, #state{socket=Socket, gid=DestGID}) -> +send_0c10(Options, #state{socket=Socket, gid=DestGID, lid=DestLID}) -> 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. %% @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 >>). %% @todo Always the same value, no idea what it's for. -%% @todo Handle the LID properly. -send_1204(#state{socket=Socket, gid=DestGID}) -> - packet_send(Socket, << 16#12040300:32, 0:160, 16#00011300:32, DestGID:32/little, 0:96, 16#20000000:32, 0:256 >>). +send_1204(#state{socket=Socket, gid=DestGID, lid=DestLID}) -> + packet_send(Socket, << 16#12040300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:96, 16#20000000:32, 0:256 >>). %% @doc Send the player's partner card. %% @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 >>). %% @doc Send the background to use for the counter. -%% @todo Handle LID properly. -send_1711(Bg, #state{socket=Socket, gid=DestGID}) -> - packet_send(Socket, << 16#17110300:32, 0:160, 16#00011300:32, DestGID:32/little, 0:64, Bg:8, 0:24 >>). +send_1711(Bg, #state{socket=Socket, gid=DestGID, lid=DestLID}) -> + packet_send(Socket, << 16#17110300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:64, Bg:8, 0:24 >>). %% @doc NPC shop request reply. -%% @todo Handle the LID properly. -send_1a02(A, B, C, D, #state{socket=Socket, gid=DestGID}) -> - packet_send(Socket, << 16#1a020300:32, 0:160, 16#00011300:32, DestGID:32/little, 0:96, +send_1a02(A, B, C, D, #state{socket=Socket, gid=DestGID, lid=DestLID}) -> + packet_send(Socket, << 16#1a020300:32, DestLID:16/little, 0:144, 16#00011300:32, DestGID:32/little, 0:96, A:16/little, B:16/little, C:16/little, D:16/little >>). %% @doc Lumilass available hairstyles/headtypes handler. -%% @todo Handle the LID properly. -send_1a03(CharUser, #state{socket=Socket, gid=DestGID}) -> +send_1a03(CharUser, #state{socket=Socket, gid=DestGID, lid=DestLID}) -> {ok, Conf} = file:consult("priv/lumilass.conf"), Character = CharUser#users.character, NbHeadtypes = proplists:get_value({headtypes, Character#characters.gender, Character#characters.race}, Conf, 0), HairstylesList = proplists:get_value({hairstyles, Character#characters.gender}, Conf), NbHairstyles = length(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 >>). %% @doc Available types handler. Enable all 16 types.