diff --git a/ebin/egs.app b/ebin/egs.app
index 46067ce..8a1e054 100644
--- a/ebin/egs.app
+++ b/ebin/egs.app
@@ -16,6 +16,8 @@
egs_proto,
psu_appearance,
psu_characters,
+ psu_party,
+ psu_npc,
psu_parser
]},
{registered, []},
diff --git a/include/psu/npc.hrl b/include/psu/npc.hrl
new file mode 100644
index 0000000..e3388c9
--- /dev/null
+++ b/include/psu/npc.hrl
@@ -0,0 +1,77 @@
+%% EGS: Erlang Game Server
+%% Copyright (C) 2010 Loic Hoguin
+%%
+%% This file is part of EGS.
+%%
+%% EGS is free software: you can redistribute it and/or modify
+%% it under the terms of the GNU General Public License as published by
+%% the Free Software Foundation, either version 3 of the License, or
+%% (at your option) any later version.
+%%
+%% EGS is distributed in the hope that it will be useful,
+%% but WITHOUT ANY WARRANTY; without even the implied warranty of
+%% MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+%% GNU General Public License for more details.
+%%
+%% You should have received a copy of the GNU General Public License
+%% along with EGS. If not, see .
+
+-record(psu_npc, {has_card, name, race, gender, class, level, appearance}).
+
+-define(NPC, [
+ %~ { 0, #psu_npc{has_card=false, name="Ethan Waber", level=+0}},
+
+ %~ { 1, #psu_npc{has_card=true, name="Hyuga Ryght", race=human, gender=male, class=hunter, level=+3,
+ %~ appearance=#flesh_appearance{voicetype=54, jacket=16#00860300, pants=16#00860301, shoes=16#00860302, ears=16#00860303, face=16#00860304, hairstyle=16#00860305}
+ %~ }},
+
+ %~ { 2, #psu_npc{has_card=true, name="Karen Erra", level=+0}}, %% normal
+ %~ { 3, #psu_npc{has_card=true, name="Leogini Berafort", level=+0}},
+ %~ { 4, #psu_npc{has_card=true, name="Lucaim Nav", level=+0}},
+ %~ { 5, #psu_npc{has_card=true, name="Maya Shidow", level=+0}},
+ %~ { 6, #psu_npc{has_card=true, name="Tonnio Rhima", level=+0}},
+
+ { 7, #psu_npc{has_card=true, name="Lou", race=cast, gender=female, class=hunter, level=+3,
+ appearance=#metal_appearance{voicetype=59, torso=16#008b1300, legs=16#008b1301, arms=16#008b1302, ears=16#008b1303, face=16#008b1304, headtype=16#008b1305}
+ }}%,
+
+ %~ { 8, #psu_npc{has_card=true, name="Mirei Mikuna", level=+0}},
+ %~ { 9, #psu_npc{has_card=true, name="Hiru Vol", level=+0}},
+ %~ {10, #psu_npc{has_card=true, name="No Vol", level=+0}},
+ %~ {11, #psu_npc{has_card=true, name="Do Vol", level=+0}},
+ %~ {12, #psu_npc{has_card=true, name="Liina Sukaya", level=+0}},
+ %~ {13, #psu_npc{has_card=true, name="Alfort Tylor", level=+0}},
+ %~ {14, #psu_npc{has_card=true, name="Obel Dallgun", level=+0}},
+ %~ {15, #psu_npc{has_card=true, name="Ethan Waber", level=+0}}, %% EP1
+ %~ {16, #psu_npc{has_card=true, name="Fulyen Curtz", level=+0}},
+ %~ {17, #psu_npc{has_card=true, name="Renvolt Magashi", level=+0}},
+ %~ {18, #psu_npc{has_card=false, name="Lumia Waber", level=+0}},
+ %~ {19, #psu_npc{has_card=true, name="Remlia Norphe", level=+0}},
+ %~ {20, #psu_npc{has_card=false, name="Clamp Maniel", level=+0}},
+ %~ {21, #psu_npc{has_card=false, name="Kanal Tomrain", level=+0}},
+
+ %~ {22, #psu_npc{has_card=false, name="Mina", race=human, gender=female, class=hunter, level=+0,
+ %~ appearance=#flesh_appearance{voicetype=87, jacket=16#009C1300, pants=16#009C1301, shoes=16#009C1302, ears=16#009C1303, face=16#009C1304, hairstyle=16#009C1305}
+ %~ }},
+
+ %~ {23, #psu_npc{has_card=true, name="Hal", level=+0}},
+ %~ {24, #psu_npc{has_card=false, name="Fulyen Curtz", level=+0}},
+ %~ {25, #psu_npc{has_card=true, name="Laia Martinez", level=+0}}, %% EP2
+ %~ {26, #psu_npc{has_card=true, name="Karen Erra", level=+0}}, %% maiden
+ %~ {27, #psu_npc{has_card=false, name="Mirei Mikuna", level=+0}},
+ %~ {28, #psu_npc{has_card=false, name="Obel Dallgun", level=+0}},
+ %~ {29, #psu_npc{has_card=false, name="Maira Klein", level=+0}},
+ %~ {30, #psu_npc{has_card=true, name="Orson Waber", level=+0}},
+ %~ {31, #psu_npc{has_card=false, name="Fulyen Curtz", level=+0}},
+ %~ {32, #psu_npc{has_card=true, name="Bruce Boyde", level=+0}},
+ %~ {33, #psu_npc{has_card=true, name="Ethan Waber", level=+0}}, %% rogue
+ %~ {34, #psu_npc{has_card=false, name="Vivienne", level=+0}},
+ %~ {35, #psu_npc{has_card=false, name="Helga", level=+0}},
+ %~ {36, #psu_npc{has_card=false, name="Hakana Kutanami", level=+0}},
+ %~ {37, #psu_npc{has_card=false, name="Liche Baratse", level=+0}},
+ %~ {38, #psu_npc{has_card=true, name="Howzer", level=+0}},
+ %~ {39, #psu_npc{has_card=true, name="Rutsu", level=+0}},
+ %~ {40, #psu_npc{has_card=true, name="Lumia Waber", level=+0}}, %% EP2
+ %~ {41, #psu_npc{has_card=true, name="Laia Martinez", level=+0}}, %% president
+ %~ {42, #psu_npc{has_card=true, name="My PM", level=+0}}
+]).
diff --git a/include/psu_npc.hrl b/include/psu_npc.hrl
deleted file mode 100644
index cd67d49..0000000
--- a/include/psu_npc.hrl
+++ /dev/null
@@ -1,65 +0,0 @@
-%% EGS: Erlang Game Server
-%% Copyright (C) 2010 Loic Hoguin
-%%
-%% This file is part of EGS.
-%%
-%% EGS is free software: you can redistribute it and/or modify
-%% it under the terms of the GNU General Public License as published by
-%% the Free Software Foundation, either version 3 of the License, or
-%% (at your option) any later version.
-%%
-%% EGS is distributed in the hope that it will be useful,
-%% but WITHOUT ANY WARRANTY; without even the implied warranty of
-%% MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-%% GNU General Public License for more details.
-%%
-%% You should have received a copy of the GNU General Public License
-%% along with EGS. If not, see .
-
--record(psu_npc, {has_card, name, level}).
-
--define(NPC, [
- { 0, #psu_npc{has_card=false, name="Ethan Waber", level=+0}},
- { 1, #psu_npc{has_card=true, name="Hyuga Ryght", level=+0}},
- { 2, #psu_npc{has_card=true, name="Karen Erra", level=+0}}, %% normal
- { 3, #psu_npc{has_card=true, name="Leogini Berafort", level=+0}},
- { 4, #psu_npc{has_card=true, name="Lucaim Nav", level=+0}},
- { 5, #psu_npc{has_card=true, name="Maya Shidow", level=+0}},
- { 6, #psu_npc{has_card=true, name="Tonnio Rhima", level=+0}},
- { 7, #psu_npc{has_card=true, name="Lou", level=+0}},
- { 8, #psu_npc{has_card=true, name="Mirei Mikuna", level=+0}},
- { 9, #psu_npc{has_card=true, name="Hiru Vol", level=+0}},
- {10, #psu_npc{has_card=true, name="No Vol", level=+0}},
- {11, #psu_npc{has_card=true, name="Do Vol", level=+0}},
- {12, #psu_npc{has_card=true, name="Liina Sukaya", level=+0}},
- {13, #psu_npc{has_card=true, name="Alfort Tylor", level=+0}},
- {14, #psu_npc{has_card=true, name="Obel Dallgun", level=+0}},
- {15, #psu_npc{has_card=true, name="Ethan Waber", level=+0}}, %% EP1
- {16, #psu_npc{has_card=true, name="Fulyen Curtz", level=+0}},
- {17, #psu_npc{has_card=true, name="Renvolt Magashi", level=+0}},
- {18, #psu_npc{has_card=false, name="Lumia Waber", level=+0}},
- {19, #psu_npc{has_card=true, name="Remlia Norphe", level=+0}},
- {20, #psu_npc{has_card=false, name="Clamp Maniel", level=+0}},
- {21, #psu_npc{has_card=false, name="Kanal Tomrain", level=+0}},
- {22, #psu_npc{has_card=false, name="Mina", level=+0}},
- {23, #psu_npc{has_card=true, name="Hal", level=+0}},
- {24, #psu_npc{has_card=false, name="Fulyen Curtz", level=+0}},
- {25, #psu_npc{has_card=true, name="Laia Martinez", level=+0}}, %% EP2
- {26, #psu_npc{has_card=true, name="Karen Erra", level=+0}}, %% maiden
- {27, #psu_npc{has_card=false, name="Mirei Mikuna", level=+0}},
- {28, #psu_npc{has_card=false, name="Obel Dallgun", level=+0}},
- {29, #psu_npc{has_card=false, name="Maira Klein", level=+0}},
- {30, #psu_npc{has_card=true, name="Orson Waber", level=+0}},
- {31, #psu_npc{has_card=false, name="Fulyen Curtz", level=+0}},
- {32, #psu_npc{has_card=true, name="Bruce Boyde", level=+0}},
- {33, #psu_npc{has_card=true, name="Ethan Waber", level=+0}}, %% rogue
- {34, #psu_npc{has_card=false, name="Vivienne", level=+0}},
- {35, #psu_npc{has_card=false, name="Helga", level=+0}},
- {36, #psu_npc{has_card=false, name="Hakana Kutanami", level=+0}},
- {37, #psu_npc{has_card=false, name="Liche Baratse", level=+0}},
- {38, #psu_npc{has_card=true, name="Howzer", level=+0}},
- {39, #psu_npc{has_card=true, name="Rutsu", level=+0}},
- {40, #psu_npc{has_card=true, name="Lumia Waber", level=+0}}, %% EP2
- {41, #psu_npc{has_card=true, name="Laia Martinez", level=+0}}, %% president
- {42, #psu_npc{has_card=true, name="My PM", level=+0}}
-]).
diff --git a/include/records.hrl b/include/records.hrl
index f1ec28d..cb2c6c8 100644
--- a/include/records.hrl
+++ b/include/records.hrl
@@ -31,7 +31,7 @@
%% @todo Probably can use a "param" or "extra" field to store the game-specific information (for things that don't need to be queried).
-record(egs_user_model, {
- id, pid, socket, state, time, character, instancepid, areatype, area, entryid, pos,
+ id, pid, socket, state, time, character, instancepid, partypid, areatype, area, entryid, pos,
%% psu specific fields
lid, setid, prev_area, prev_entryid,
%% temporary fields
@@ -48,15 +48,32 @@
%% @doc Character appearance data structure, flesh version.
--record(flesh_appearance, {voicetype, voicepitch, jacket, pants, shoes, ears, face, hairstyle, jacketcolor, pantscolor, shoescolor,
- lineshieldcolor, badge, eyebrows, eyelashes, eyesgroup, eyes, bodysuit, eyescolory, eyescolorx, lipsintensity, lipscolory, lipscolorx,
- skincolor, hairstylecolory, hairstylecolorx, proportion, proportionboxx, proportionboxy, faceboxx, faceboxy}).
+-record(flesh_appearance, {
+ voicetype, voicepitch=127,
+ jacket, pants, shoes, ears, face, hairstyle,
+ jacketcolor=0, pantscolor=0, shoescolor=0, lineshieldcolor=0, badge=0,
+ eyebrows=0, eyelashes=0, eyesgroup=0, eyes=0,
+ bodysuit=0,
+ eyescolory=32767, eyescolorx=0,
+ lipsintensity=32767, lipscolory=32767, lipscolorx=0,
+ skincolor=65535,
+ hairstylecolory=32767, hairstylecolorx=0,
+ proportion=65535, proportionboxx=65535, proportionboxy=65535,
+ faceboxx=65535, faceboxy=65535
+}).
%% @doc Character appearance data structure, metal version.
-
--record(metal_appearance, {voicetype, voicepitch, torso, legs, arms, ears, face, headtype, maincolor, lineshieldcolor,
- eyebrows, eyelashes, eyesgroup, eyes, eyescolory, eyescolorx, bodycolor, subcolor, hairstylecolory, hairstylecolorx,
- proportion, proportionboxx, proportionboxy, faceboxx, faceboxy}).
+-record(metal_appearance, {
+ voicetype, voicepitch=127,
+ torso, legs, arms, ears, face, headtype,
+ maincolor=0, lineshieldcolor=0,
+ eyebrows=0, eyelashes=0, eyesgroup=0, eyes=0,
+ eyescolory=32767, eyescolorx=0,
+ bodycolor=65535, subcolor=196607,
+ hairstylecolory=32767, hairstylecolorx=0,
+ proportion=65535, proportionboxx=65535, proportionboxy=65535,
+ faceboxx=65535, faceboxy=65535
+}).
%% @doc Character options data structure.
@@ -76,6 +93,7 @@
gid,
type=white,
slot,
+ npcid=0,
name,
race,
gender,
diff --git a/p/packet0a04.bin b/p/packet0a04.bin
new file mode 100644
index 0000000..74695ad
Binary files /dev/null and b/p/packet0a04.bin differ
diff --git a/p/packet1601.bin b/p/packet1601.bin
new file mode 100644
index 0000000..d5d8d00
Binary files /dev/null and b/p/packet1601.bin differ
diff --git a/src/egs_user_model.erl b/src/egs_user_model.erl
index 7e31a56..87eb455 100644
--- a/src/egs_user_model.erl
+++ b/src/egs_user_model.erl
@@ -94,12 +94,16 @@ handle_call({read, ID}, _From, State) ->
%% @todo state = undefined | {wait_for_authentication, Key} | authenticated | online
handle_call({select, all}, _From, State) ->
- List = do(qlc:q([X || X <- mnesia:table(?TABLE), X#?TABLE.state =:= online])),
+ List = do(qlc:q([X || X <- mnesia:table(?TABLE),
+ X#?TABLE.pid /= undefined,
+ X#?TABLE.state =:= online
+ ])),
{reply, {ok, List}, State};
handle_call({select, {neighbors, User}}, _From, State) ->
List = do(qlc:q([X || X <- mnesia:table(?TABLE),
X#?TABLE.id /= User#?TABLE.id,
+ X#?TABLE.pid /= undefined,
X#?TABLE.state =:= online,
X#?TABLE.instancepid =:= User#?TABLE.instancepid,
X#?TABLE.area =:= User#?TABLE.area
diff --git a/src/psu/psu_characters.erl b/src/psu/psu_characters.erl
index a35a0d8..1236b1d 100644
--- a/src/psu/psu_characters.erl
+++ b/src/psu/psu_characters.erl
@@ -29,29 +29,36 @@
%% Only contains the actually saved data, not the stats and related information.
character_tuple_to_binary(Tuple) ->
- #characters{name=Name, race=Race, gender=Gender, class=Class, appearance=Appearance,
+ #characters{type=Type, name=Name, race=Race, gender=Gender, class=Class, appearance=Appearance,
mainlevel=Level, blastbar=BlastBar, luck=Luck, money=Money, playtime=PlayTime} = Tuple,
#level{number=LV, exp=EXP} = Level,
RaceBin = race_atom_to_binary(Race),
GenderBin = gender_atom_to_binary(Gender),
ClassBin = class_atom_to_binary(Class),
AppearanceBin = psu_appearance:tuple_to_binary(Race, Appearance),
+ FooterBin = case Type of
+ npc ->
+ << 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32,
+ 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32,
+ 16#4e4f4630:32, 16#08000000:32, 0:32, 0:32, 16#4e454e44:32 >>;
+ _ -> %% @todo Handle classes.
+ << 0:160,
+ 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32,
+ 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32 >>
+ end,
<< Name/binary, RaceBin:8, GenderBin:8, ClassBin:8, AppearanceBin/binary, LV:32/little-unsigned-integer, BlastBar:16/little-unsigned-integer,
- Luck:8, 0:40, EXP:32/little-unsigned-integer, 0:32, Money:32/little-unsigned-integer, PlayTime:32/little-unsigned-integer, 0:160,
- % then classes hardcoded for now @todo
- 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32,
- 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32 >>.
+ Luck:8, 0:40, EXP:32/little-unsigned-integer, 0:32, Money:32/little-unsigned-integer, PlayTime:32/little-unsigned-integer, FooterBin/binary >>.
%% @doc Convert a character tuple into a binary to be sent to clients.
%% Contains everything from character_tuple_to_binary/1 along with location, stats, SE and more.
-%% @todo One of the two QuestID lists has a different use. No idea what though.
+%% @todo One of the two QuestID lists has a different use. No idea what though. The second is probably the previous area.
%% @todo The second StatsBin seems unused. Not sure what it's for.
%% @todo Find out what the big block of 0 is at the end.
%% @todo The value before IntDir seems to be the player's current animation. 01 stand up, 08 ?, 17 normal sit
character_user_to_binary(User) ->
#egs_user_model{id=CharGID, lid=CharLID, character=Character, pos=#pos{x=X, y=Y, z=Z, dir=Dir}, area={psu_area, QuestID, ZoneID, MapID}, entryid=EntryID} = User,
- #characters{mainlevel=Level, stats=Stats, se=SE, currenthp=CurrentHP, maxhp=MaxHP} = Character,
+ #characters{type=Type, mainlevel=Level, stats=Stats, se=SE, currenthp=CurrentHP, maxhp=MaxHP} = Character,
#level{number=LV} = Level,
CharBin = psu_characters:character_tuple_to_binary(Character),
StatsBin = psu_characters:stats_tuple_to_binary(Stats),
@@ -59,12 +66,15 @@ character_user_to_binary(User) ->
EXPNextLevel = 100,
EXPPreviousLevel = 0,
IntDir = trunc(Dir * 182.0416),
- << 16#00001200:32, CharGID:32/little-unsigned-integer, 0:64, CharLID:32/little-unsigned-integer, 16#0000ffff:32, QuestID:32/little-unsigned-integer,
+ TypeID = case Type of npc -> 16#00001d00; _ -> 16#00001200 end,
+ NPCStuff = case Type of npc -> 16#01ff0700; _ -> 16#0000ffff end,
+ << TypeID:32, CharGID:32/little-unsigned-integer, 0:64, CharLID:32/little-unsigned-integer, NPCStuff:32, QuestID:32/little-unsigned-integer,
ZoneID:32/little-unsigned-integer, MapID:32/little-unsigned-integer, EntryID:32/little-unsigned-integer,
16#0100:16, IntDir:16/little-unsigned-integer, X:32/little-float, Y:32/little-float, Z:32/little-float, 0:64,
QuestID:32/little-unsigned-integer, ZoneID:32/little-unsigned-integer, MapID:32/little-unsigned-integer, EntryID:32/little-unsigned-integer,
CharBin/binary, EXPNextLevel:32/little-unsigned-integer, EXPPreviousLevel:32/little-unsigned-integer, MaxHP:32/little-unsigned-integer, % not sure if this one is current or max
- StatsBin/binary, 0:32, SEBin/binary, 0:32, LV:32/little-unsigned-integer, StatsBin/binary, CurrentHP:32/little-unsigned-integer, MaxHP:32/little-unsigned-integer, 0:2304 >>.
+ StatsBin/binary, 0:32, SEBin/binary, 0:32, LV:32/little-unsigned-integer, StatsBin/binary, CurrentHP:32/little-unsigned-integer, MaxHP:32/little-unsigned-integer,
+ 0:1344, 16#0000803f:32, 0:64, 16#0000803f:32, 0:64, 16#0000803f:32, 0:64, 16#0000803f:32, 0:64, 16#0000803f:32, 0:160, 16#0000803f:32, 0:352 >>.
%% @doc Convert a class atom into a binary to be sent to clients.
diff --git a/src/psu/psu_game.erl b/src/psu/psu_game.erl
index 61c8ccb..65e8e63 100644
--- a/src/psu/psu_game.erl
+++ b/src/psu/psu_game.erl
@@ -24,7 +24,7 @@
-include("include/records.hrl").
-include("include/maps.hrl").
-include("include/missions.hrl").
--include("include/psu_npc.hrl").
+-include("include/psu/npc.hrl").
-define(OPTIONS, [binary, {active, false}, {reuseaddr, true}, {certfile, "priv/ssl/servercert.pem"}, {keyfile, "priv/ssl/serverkey.pem"}, {password, "alpha"}]).
@@ -243,7 +243,7 @@ counter_load(QuestID, ZoneID, MapID, EntryID) ->
send_0c00(16#7fffffff),
send_020e(QuestFile),
send_0a05(),
- send_010d(User),
+ send_010d(User#egs_user_model{lid=0}),
send_0200(mission),
send_020f(ZoneFile, 0, 16#ff),
send_0205(0, 0, 0, 0),
@@ -256,7 +256,7 @@ counter_load(QuestID, ZoneID, MapID, EntryID) ->
send_1206(),
send_1207(),
send_1212(),
- send_0201(User),
+ send_0201(User#egs_user_model{lid=0}),
send_0a06(),
send_0208(),
send_0236().
@@ -379,7 +379,7 @@ area_load(AreaType, IsStart, SetID, OldUser, User, QuestFile, ZoneFile, AreaName
send_0111(6, 0);
true -> ignore
end,
- send_010d(User),
+ send_010d(User#egs_user_model{lid=0}),
send_0200(AreaType),
send_020f(ZoneFile, SetID, SeasonID);
true -> ignore
@@ -417,14 +417,41 @@ area_load(AreaType, IsStart, SetID, OldUser, User, QuestFile, ZoneFile, AreaName
send_1309();
true -> ignore
end,
- send_0201(User),
+ send_0201(User#egs_user_model{lid=0}),
if ZoneChange =:= true ->
send_0a06();
true -> ignore
end,
send_0233(SpawnList),
send_0208(),
- send_0236().
+ send_0236(),
+ if User#egs_user_model.partypid =/= undefined, AreaType =:= mission ->
+ {ok, NPCList} = psu_party:get_npc(User#egs_user_model.partypid),
+ npc_load(User, NPCList);
+ true -> ok
+ end.
+
+%% @todo Make NPC hide in lobbies but show-up in missions.
+%% @todo Don't change the NPC info unless you are the leader!
+npc_load(_Leader, []) ->
+ ok;
+npc_load(Leader, [{PartyPos, NPCGID}|NPCList]) ->
+ {ok, OldNPCUser} = egs_user_model:read(NPCGID),
+ #egs_user_model{instancepid=InstancePid, area=Area, entryid=EntryID, pos=Pos} = Leader,
+ NPCUser = OldNPCUser#egs_user_model{lid=PartyPos, instancepid=InstancePid, areatype=mission, area=Area, entryid=EntryID, pos=Pos},
+ io:format("~p", [NPCUser]),
+ %% @todo This one on mission end/abort?
+ %~ OldNPCUser#egs_user_model{lid=PartyPos, instancepid=undefined, areatype=AreaType, area={psu_area, 0, 0, 0}, entryid=0, pos={pos, 0.0, 0.0, 0.0, 0}}
+ egs_user_model:write(NPCUser),
+ send_010d(NPCUser),
+ send_0201(NPCUser),
+ send_0215(0),
+ send_0a04(NPCUser#egs_user_model.id),
+ send_1004(npc_mission, NPCUser, PartyPos),
+ send_100f((NPCUser#egs_user_model.character)#characters.npcid, PartyPos),
+ send_1601(),
+ send_1016(PartyPos),
+ npc_load(Leader, NPCList).
%% @doc Game's main loop.
%% @todo We probably don't want to send a keepalive packet unnecessarily.
@@ -435,8 +462,8 @@ loop(SoFar) ->
GID = get(gid),
send(<< A/binary, 16#00011300:32, B/binary, 16#00011300:32, GID:32/little-unsigned-integer, C/binary >>),
?MODULE:loop(SoFar);
- {psu_chat, ChatGID, ChatName, ChatModifiers, ChatMessage} ->
- send_0304(ChatGID, ChatName, ChatModifiers, ChatMessage),
+ {psu_chat, ChatTypeID, ChatGID, ChatName, ChatModifiers, ChatMessage} ->
+ send_0304(ChatTypeID, ChatGID, ChatName, ChatModifiers, ChatMessage),
?MODULE:loop(SoFar);
{psu_keepalive} ->
egs_proto:send_keepalive(get(socket)),
@@ -534,9 +561,10 @@ handle(16#0102, _) ->
%% @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 We must also handle here the NPC characters. PartyPos can be used for that, and more info in unknown values maybe too?
handle(16#0105, Data) ->
- << _:32, A:32/little-unsigned-integer, ItemID:8, Action:8, _:8, B:8, C:32/little-unsigned-integer, _/bits >> = Data,
- log("0105 action ~b item ~b (~b ~b ~b)", [Action, ItemID, A, B, C]),
+ << _:32, PartyPos:32/little-unsigned-integer, ItemID:8, Action:8, _:8, A:8, B:32/little-unsigned-integer, _/bits >> = Data,
+ log("0105 action ~b item ~b partypos ~b (~b ~b)", [Action, ItemID, PartyPos, A, B]),
GID = get(gid),
Category = case ItemID of
% units would be 8, traps would be 12
@@ -569,11 +597,11 @@ handle(16#0105, Data) ->
_ -> {ok, File} = file:read_file(Filename)
end,
send(<< 16#01050300:32, 0:64, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer,
- 0:64, GID:32/little-unsigned-integer, A:32/little-unsigned-integer, ItemID, Action, Category, B, C:32/little-unsigned-integer,
+ 0:64, GID:32/little-unsigned-integer, PartyPos:32/little-unsigned-integer, ItemID, Action, Category, A, B:32/little-unsigned-integer,
File/binary >>);
2 -> % unequip item
send(<< 16#01050300: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, A:32/little-unsigned-integer, ItemID, Action, Category, B, C:32/little-unsigned-integer >>);
+ 0:64, GID:32/little-unsigned-integer, PartyPos:32/little-unsigned-integer, ItemID, Action, Category, A, B:32/little-unsigned-integer >>);
5 -> % drop item
ignore;
_ ->
@@ -592,6 +620,7 @@ handle(16#010a, Data) ->
%% @doc Character death, and more, handler. Warp to 4th floor for now.
%% @todo Recover from death correctly.
+%% @todo A is probably PartyPos or LID.
handle(16#0110, Data) ->
<< _:32, A:32/little-unsigned-integer, B:32/little-unsigned-integer, C:32/little-unsigned-integer >> = Data,
case B of
@@ -599,6 +628,8 @@ handle(16#0110, Data) ->
send_0113();
3 -> % type change
log("changed type to ~b", [C]);
+ 4 -> % related to npc death, ignore for now
+ ignore;
7 -> % player death: if the player has a scape, use it! otherwise red screen @todo Right now we force revive and don't reset the HP.
% @todo send_0115(get(gid), 16#ffffffff, LV=1, EXP=idk, Money=1000), % apparently sent everytime you die...
% use scape
@@ -639,8 +670,15 @@ handle(16#021f, << Uni:32/little-unsigned-integer, _/bits >>) ->
% 0220
% force reloading the character and data files (hack)
{ok, User} = egs_user_model:read(get(gid)),
+ if User#egs_user_model.partypid =:= undefined ->
+ ignore;
+ true ->
+ %% @todo Replace stop by leave when leaving stops the party correctly when nobody's there anymore.
+ %~ psu_party:leave(User#egs_user_model.partypid, User#egs_user_model.id)
+ psu_party:stop(User#egs_user_model.partypid)
+ end,
Area = User#egs_user_model.area,
- NewRow = User#egs_user_model{area=Area#psu_area{questid=1120000, zoneid=undefined}},
+ NewRow = User#egs_user_model{partypid=undefined, area=Area#psu_area{questid=1120000, zoneid=undefined}},
egs_user_model:write(NewRow),
area_load(Area#psu_area.questid, Area#psu_area.zoneid, Area#psu_area.mapid, User#egs_user_model.entryid)
end;
@@ -654,20 +692,32 @@ handle(16#0302, _) ->
%% We must take extra precautions to handle different versions of the game correctly.
%% Disregard the name sent by the server in later versions of the game. 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 and forces it to talk like in the tutorial mission it seems FromTypeID, FromGID and Name are both 0.
handle(16#0304, Data) ->
- {ok, User} = egs_user_model:read(get(gid)),
case get(version) of
0 -> % AOTI v2.000
- << _:64, Modifiers:128/bits, Message/bits >> = Data;
+ << FromTypeID:32/unsigned-integer, FromGID:32/little-unsigned-integer, Modifiers:128/bits, Message/bits >> = Data;
_ -> % Above
- << _:64, Modifiers:128/bits, _:512, Message/bits >> = Data
+ << FromTypeID:32/unsigned-integer, FromGID:32/little-unsigned-integer, Modifiers:128/bits, _:512, Message/bits >> = Data
end,
+
+ UserGID = get(gid),
+ GID = if UserGID =:= FromGID ->
+ UserGID;
+ true ->
+ %% @todo Check that FromGID is an NPC in the UserGID's party; that UserGID is the party leader; that the message is using party chat.
+ FromGID
+ end,
+
+ {ok, User} = egs_user_model:read(GID),
+
[LogName|_] = re:split((User#egs_user_model.character)#characters.name, "\\0\\0", [{return, binary}]),
[TmpMessage|_] = re:split(Message, "\\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}])]]),
{ok, List} = egs_user_model:select(all),
- lists:foreach(fun(X) -> X#egs_user_model.pid ! {psu_chat, get(gid), (User#egs_user_model.character)#characters.name, Modifiers, Message} end, List);
+ lists:foreach(fun(X) -> X#egs_user_model.pid ! {psu_chat, FromTypeID, GID, (User#egs_user_model.character)#characters.name, Modifiers, Message} end, List);
%% @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.
@@ -717,17 +767,36 @@ handle(16#0812, _) ->
%% @doc NPC invite.
%% @todo Also happening a 1506 -> 1507? Only on first selection from menu.
-%% @todo Apparently Unknown is ffffffff.
-%% @todo Also sent a 101a (NPC:16, PartyPos:16, ffffffff). Not sure about PartyPos.
-%% @todo Replace 1 by the actual character level.
-%% @todo PartyPos isn't handled yet, it's always 5.
+%% @todo Apparently Unknown is always ffffffff.
+%% @todo Also at the end send a 101a (NPC:16, PartyPos:16, ffffffff). Not sure about PartyPos.
+%% @todo Probably needs to make the NPC show up if he's invited while in-mission.
handle(16#0813, Data) ->
+ GID = get(gid),
+ {ok, User} = egs_user_model:read(GID),
+ %% Create NPC.
<< _Unknown:32, NPCid:32/little-unsigned-integer >> = Data,
- NPC = proplists:get_value(NPCid, ?NPC),
- log("invited npc ~s", [NPC#psu_npc.name]),
- PartyPos = 5,
- send_022c(0, 2),
- send_1004(NPCid, NPCid, NPC#psu_npc.name, 1 + NPC#psu_npc.level, PartyPos, 0, 0, 0, 0);
+ log("invited npcid ~b", [NPCid]),
+ TmpNPCUser = psu_npc:user_init(NPCid, ((User#egs_user_model.character)#characters.mainlevel)#level.number),
+ %% Create and join party.
+ %% @todo Check if party already exists.
+ {ok, PartyPid} = psu_party:start_link(GID),
+ {ok, PartyPos} = psu_party:join(PartyPid, npc, TmpNPCUser#egs_user_model.id),
+ NPCUser = TmpNPCUser#egs_user_model{lid=PartyPos, partypid=PartyPid},
+ egs_user_model:write(NPCUser),
+ egs_user_model:write(User#egs_user_model{partypid=PartyPid}),
+ %% Send stuff.
+ Character = NPCUser#egs_user_model.character,
+ SentNPCCharacter = Character#characters{gid=NPCid},
+ SentNPCUser = NPCUser#egs_user_model{id=NPCid, character=SentNPCCharacter},
+ %% @todo send_022c(0, 2),
+ send_1004(npc_invite, SentNPCUser, PartyPos),
+ send_101a(NPCid, PartyPos);
+
+%% @todo Used in the tutorial. Not sure what it does. Give an item (the PA) maybe?
+handle(16#0a09, Data) ->
+ log("~p", [Data]),
+ GID = get(gid),
+ send(<< 16#0a090300:32, 0:32, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, 16#00003300:32, 0:32 >>);
%% @doc Item description request.
%% @todo Send something other than just "dammy".
@@ -776,7 +845,6 @@ handle(16#0c0e, _) ->
end;
%% @doc Counter available mission list request handler.
-%% @todo Temporarily allow rare mission and LL all difficulties to all players.
handle(16#0c0f, _) ->
{ok, User} = egs_user_model:read(get(gid)),
[{quests, _}, {bg, _}, {options, Options}] = proplists:get_value(User#egs_user_model.entryid, ?COUNTERS),
@@ -1077,10 +1145,11 @@ send(Packet) ->
send_010d(User) ->
GID = get(gid),
CharGID = User#egs_user_model.id,
- << _:640, CharBin/bits >> = psu_characters:character_user_to_binary(User#egs_user_model{lid=0}),
+ CharLID = User#egs_user_model.lid,
+ << _:640, CharBin/bits >> = psu_characters:character_user_to_binary(User),
send(<< 16#010d0300:32, 0:160, 16#00011300:32, GID:32/little-unsigned-integer, 0:64,
1:32/little-unsigned-integer, 0:32, 16#00000300:32, 16#ffff0000:32, 0:32, CharGID:32/little-unsigned-integer,
- 0:192, CharGID:32/little-unsigned-integer, 0:32, 16#ffffffff:32, CharBin/binary >>).
+ 0:192, CharGID:32/little-unsigned-integer, CharLID:32/little-unsigned-integer, 16#ffffffff:32, CharBin/binary >>).
%% @todo Possibly related to 010d. Just send seemingly safe values.
send_0111(A, B) ->
@@ -1125,7 +1194,7 @@ send_0200(ZoneType) ->
send_0201(User) ->
GID = get(gid),
CharGID = User#egs_user_model.id,
- CharBin = psu_characters:character_user_to_binary(User#egs_user_model{lid=0}),
+ CharBin = psu_characters:character_user_to_binary(User),
IsGM = 0,
OnlineStatus = 0,
GameVersion = 0,
@@ -1255,10 +1324,10 @@ send_0236() ->
send(header(16#0236)).
%% @doc Send a chat command. Handled differently at v2.0000 and all versions starting somewhere above that.
-send_0304(FromGID, FromName, Modifiers, Message) ->
+send_0304(FromTypeID, FromGID, FromName, Modifiers, Message) ->
case get(version) of
- 0 -> send(<< 16#03040300:32, 0:288, 16#00001200:32, FromGID:32/little-unsigned-integer, Modifiers:128/bits, Message/bits >>);
- _ -> send(<< 16#03040300:32, 0:288, 16#00001200:32, FromGID:32/little-unsigned-integer, Modifiers:128/bits, FromName:512/bits, Message/bits >>)
+ 0 -> send(<< 16#03040300:32, 0:288, FromTypeID:32/unsigned-integer, FromGID:32/little-unsigned-integer, Modifiers:128/bits, Message/bits >>);
+ _ -> send(<< 16#03040300:32, 0:288, FromTypeID:32/unsigned-integer, FromGID:32/little-unsigned-integer, Modifiers:128/bits, FromName:512/bits, Message/bits >>)
end.
%% @todo Force send a new player location. Used for warps.
@@ -1271,6 +1340,12 @@ send_0503(#pos{x=PrevX, y=PrevY, z=PrevZ, dir=_}) ->
16#1000:16, IntDir:16/little-unsigned-integer, PrevX:32/little-float, PrevY:32/little-float, PrevZ:32/little-float, X:32/little-float, Y:32/little-float, Z:32/little-float,
QuestID:32/little-unsigned-integer, ZoneID:32/little-unsigned-integer, MapID:32/little-unsigned-integer, EntryID:32/little-unsigned-integer, 1:32/little-unsigned-integer >>).
+%% @todo NPC inventory. Guessing it's only for NPC characters...
+send_0a04(NPCGID) ->
+ GID = get(gid),
+ {ok, Bin} = file:read_file("p/packet0a04.bin"),
+ send(<< 16#0a040300:32, 0:32, 16#00001d00:32, NPCGID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, Bin/binary >>).
+
%% @todo Inventory related. No idea what it does.
send_0a05() ->
send(header(16#0a05)).
@@ -1364,23 +1439,38 @@ send_0d05() ->
%% @todo Add a character (NPC or real) to the party members on the right of the screen.
%% @todo NPCid is 65535 for normal characters.
-%% @todo Apparently the 4 location ids are set to 0 when inviting an NPC in the lobby
-send_1004(GID, NPCid, Name, Level, PartyPos, QuestID, ZoneID, MapID, EntryID) ->
- UCS2Name = << << X:8, 0:8 >> || X <- Name >>,
- PaddingSize = (64 - byte_size(UCS2Name)) * 8,
- send(<< (header(16#1004))/binary, 0:32, %% first 0:32, should be 0:16, something:16
- GID:32/little-unsigned-integer, 0:64, UCS2Name/binary, 0:PaddingSize,
+%% @todo Apparently the 4 location ids are set to 0 when inviting an NPC in the lobby - NPCs have their location set to 0 when in lobby; also odd value before PartyPos related to missions
+%% @todo Not sure about LID. But seems like it.
+send_1004(Type, User, PartyPos) ->
+ io:format("~p ~p", [User, PartyPos]),
+
+ [TypeID, LID, SomeFlag] = case Type of
+ npc_mission -> [16#00001d00, PartyPos, 2];
+ npc_invite -> [0, 16#ffffffff, 3];
+ _ -> 1 %% seems to be for players
+ end,
+
+ #egs_user_model{id=GID, character=Character, area={psu_area, QuestID, ZoneID, MapID}, entryid=EntryID} = User,
+ #characters{npcid=NPCid, name=Name, mainlevel=MainLevel} = Character,
+ Level = MainLevel#level.number,
+ send(<< (header(16#1004))/binary, TypeID:32,
+ GID:32/little-unsigned-integer, 0:64, Name/binary,
Level:16/little-unsigned-integer, 16#ffff:16,
- 16#0301:16, PartyPos:8, 1, %% 03 before PartyPos, sometimes is 02 too?
+ SomeFlag, 1, PartyPos:8, 1,
NPCid:16/little-unsigned-integer, 0:16,
- %% over 512 it's sometimes just full of 0s
+
+ %% Odd unknown values. PA related? No idea. Values on invite, 0 in-mission.
16#00001f08:32, 0:32, 16#07000000:32,
16#04e41f08:32, 0:32, 16#01000000:32,
16#64e41f08:32, 0:32, 16#02000000:32,
16#64e41f08:32, 0:32, 16#03000000:32,
- 16#64e41f08:32, 0:32, 16#12000000:32, 16#24e41f08:32,
+ 16#64e41f08:32, 0:32, 16#12000000:32,
+ 16#24e41f08:32,
+
QuestID:32/little-unsigned-integer, ZoneID:32/little-unsigned-integer, MapID:32/little-unsigned-integer, EntryID:32/little-unsigned-integer,
- 16#ffffffff:32, 0:64, 16#01000000:32, 16#01000000:32, %% different values here too
+ LID:32,
+ 0:64,
+ 16#01000000:32, 16#01000000:32, %% @todo first is current hp, second is max hp
0:608 >>).
%% @todo Figure out what the packet is.
@@ -1426,6 +1516,10 @@ send_1015(QuestID) ->
send_1016(PartyPos) ->
send(<< (header(16#1016))/binary, PartyPos:32/little-unsigned-integer >>).
+%% @todo No idea.
+send_101a(NPCid, PartyPos) ->
+ send(<< (header(16#101a))/binary, NPCid:16/little-unsigned-integer, PartyPos:16/little-unsigned-integer, 16#ffffffff:32 >>).
+
%% @todo Totally unknown.
send_1020() ->
send(header(16#1020)).
@@ -1514,6 +1608,11 @@ send_1501() ->
send_1512() ->
send(<< (header(16#1512))/binary, 0:46080 >>).
+%% @todo NPC related packet, sent when there's an NPC in the area.
+send_1601() ->
+ {ok, Bin} = file:read_file("p/packet1601.bin"),
+ send(<< (header(16#1601))/binary, Bin/binary >>).
+
%% @doc Send the player's NPC and PM information.
%% @todo Do we really want to give all NPCs to everyone? Probably.
%% @todo The value 4 is the card priority. Find what 3 is. When sending, the first 0 is an unknown value.
diff --git a/src/psu/psu_npc.erl b/src/psu/psu_npc.erl
new file mode 100644
index 0000000..3b04785
--- /dev/null
+++ b/src/psu/psu_npc.erl
@@ -0,0 +1,37 @@
+%% EGS: Erlang Game Server
+%% Copyright (C) 2010 Loic Hoguin
+%%
+%% This file is part of EGS.
+%%
+%% EGS is free software: you can redistribute it and/or modify
+%% it under the terms of the GNU General Public License as published by
+%% the Free Software Foundation, either version 3 of the License, or
+%% (at your option) any later version.
+%%
+%% EGS is distributed in the hope that it will be useful,
+%% but WITHOUT ANY WARRANTY; without even the implied warranty of
+%% MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+%% GNU General Public License for more details.
+%%
+%% You should have received a copy of the GNU General Public License
+%% along with EGS. If not, see .
+
+-module(psu_npc).
+-compile(export_all).
+
+-include("include/records.hrl").
+-include("include/psu/npc.hrl").
+
+%% @todo Improve the NPC handling. Handle more than Lou.
+%% @todo Handle stats, experience, based on level.
+%% @todo Level shouldn't go below 1 or above 200.
+user_init(NPCid, BaseLevel) ->
+ NPCGID = 16#ff000000 + mnesia:dirty_update_counter(counters, npcgid, 1),
+ Settings = proplists:get_value(NPCid, ?NPC),
+ TmpUCS2Name = << << X:8, 0:8 >> || X <- Settings#psu_npc.name >>,
+ PaddingSize = 8 * (64 - byte_size(TmpUCS2Name)),
+ UCS2Name = << TmpUCS2Name/binary, 0:PaddingSize >>,
+ #psu_npc{race=Race, gender=Gender, class=Class, level=LevelDiff, appearance=Appearance} = Settings,
+ Character = #characters{gid=NPCGID, slot=0, type=npc, npcid=NPCid, name=UCS2Name, race=Race, gender=Gender, class=Class, appearance=Appearance,
+ mainlevel={level, BaseLevel + LevelDiff, 0}, blastbar=0, luck=2, money=0, playtime=0, stats={stats, 0, 0, 0, 0, 0, 0, 0}, se=[], currenthp=100, maxhp=100},
+ #egs_user_model{id=NPCGID, state=online, character=Character, areatype=lobby, area={psu_area, 0, 0, 0}, entryid=0, pos={pos, 0.0, 0.0, 0.0, 0.0}}.
diff --git a/src/psu/psu_party.erl b/src/psu/psu_party.erl
new file mode 100644
index 0000000..70221d8
--- /dev/null
+++ b/src/psu/psu_party.erl
@@ -0,0 +1,122 @@
+%% @author Loïc Hoguin
+%% @copyright 2010 Loïc Hoguin.
+%% @doc Party gen_server.
+%%
+%% This file is part of EGS.
+%%
+%% EGS is free software: you can redistribute it and/or modify
+%% it under the terms of the GNU General Public License as published by
+%% the Free Software Foundation, either version 3 of the License, or
+%% (at your option) any later version.
+%%
+%% EGS is distributed in the hope that it will be useful,
+%% but WITHOUT ANY WARRANTY; without even the implied warranty of
+%% MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+%% GNU General Public License for more details.
+%%
+%% You should have received a copy of the GNU General Public License
+%% along with EGS. If not, see .
+
+-module(psu_party).
+-behavior(gen_server).
+-export([start_link/1, stop/1, join/3, leave/2, get_instance/1, set_instance/2, remove_instance/1, get_npc/1]). %% API.
+-export([init/1, handle_call/3, handle_cast/2, handle_info/2, terminate/2, code_change/3]). %% gen_server.
+
+-record(state, {free_spots, users, instancepid}).
+
+%% API
+
+%% @spec start_link() -> {ok,Pid::pid()}
+start_link(UserID) ->
+ gen_server:start_link(?MODULE, [UserID], []).
+
+%% @spec stop() -> stopped
+stop(PartyPid) ->
+ gen_server:call(PartyPid, stop).
+
+%% @doc PlayerType is either player or npc.
+join(PartyPid, PlayerType, UserID) ->
+ gen_server:call(PartyPid, {join, PlayerType, UserID}).
+
+leave(PartyPid, UserID) ->
+ gen_server:cast(PartyPid, {leave, UserID}).
+
+get_instance(PartyPid) ->
+ gen_server:call(PartyPid, get_instance).
+
+set_instance(PartyPid, InstancePid) ->
+ gen_server:cast(PartyPid, {set_instance, InstancePid}).
+
+remove_instance(PartyPid) ->
+ gen_server:cast(PartyPid, remove_instance).
+
+%% @doc Returns a list of NPC UserID.
+get_npc(PartyPid) ->
+ gen_server:call(PartyPid, get_npc).
+
+%% gen_server
+
+init([UserID]) ->
+ error_logger:info_report("a psu_party has been started"),
+ {ok, {state, [1,2,3,4,5], [{0, leader, UserID}], undefined}}. %% 0 is party leader
+
+%% @todo Probably want to broadcast to other players that you joined the party.
+%% @todo Handle party passwords.
+handle_call({join, PlayerType, UserID}, _From, State) ->
+ List = case PlayerType of
+ npc -> lists:reverse(State#state.free_spots);
+ _ -> State#state.free_spots
+ end,
+ case List of
+ [] ->
+ {reply, {error, full}, State};
+ [Spot|FreeSpots] ->
+ Users = State#state.users,
+ SavedFreeSpots = case PlayerType of
+ npc -> lists:reverse(FreeSpots);
+ _ -> FreeSpots
+ end,
+ {reply, {ok, Spot}, State#state{free_spots=SavedFreeSpots, users=[{Spot, PlayerType, UserID}|Users]}}
+ end;
+
+handle_call(get_instance, _From, State) ->
+ {reply, {ok, State#state.instancepid}, State};
+
+handle_call(get_npc, _From, State) ->
+ List = [{Spot, UserID} || {Spot, PlayerType, UserID} <- State#state.users, PlayerType =:= npc],
+ {reply, {ok, List}, State};
+
+%% @todo Delete npc users when the party stops.
+handle_call(stop, _From, State) ->
+ {stop, normal, stopped, State};
+
+handle_call(_Request, _From, State) ->
+ {reply, ignored, State}.
+
+%% @todo Probably want to broadcast to other players that you left the party.
+%% @todo Stop the party when it becomes empty.
+%% @todo Delete npc users when the leader leaves.
+%% @todo Give leader to someone else.
+handle_cast({leave, _UserID}, State) ->
+ %% @todo Do it.
+ {noreply, State};
+
+%% @todo Probably want to broadcast to other players that an instance started.
+handle_cast({set_instance, InstancePid}, State) ->
+ {noreply, State#state{instancepid=InstancePid}};
+
+%% @todo Probably want to broadcast to other players that an instance stopped.
+handle_cast(remove_instance, State) ->
+ {noreply, State#state{instancepid=undefined}};
+
+handle_cast(_Msg, State) ->
+ {noreply, State}.
+
+handle_info(_Info, State) ->
+ {noreply, State}.
+
+terminate(_Reason, _State) ->
+ ok.
+
+code_change(_OldVsn, State, _Extra) ->
+ {ok, State}.