Added psu_instance, a process to handle missions and more later.

This commit is contained in:
Loïc Hoguin 2010-07-23 21:46:03 +02:00
parent 1b9d275b83
commit 473f7f1ae6
10 changed files with 271 additions and 250 deletions

View File

@ -13,10 +13,10 @@
psu_game, psu_game,
psu_login, psu_login,
psu_patch, psu_patch,
psu_instance,
egs_proto, egs_proto,
psu_appearance, psu_appearance,
psu_characters, psu_characters,
psu_missions,
psu_parser psu_parser
]}, ]},
{registered, []}, {registered, []},

View File

@ -20,13 +20,11 @@
-define(PATCH_PORT_JP, 11030). -define(PATCH_PORT_JP, 11030).
-define(PATCH_PORT_US, 11230). -define(PATCH_PORT_US, 11230).
-define(PATCH_LISTEN_OPTIONS, [binary, {send_timeout, 5000}, {packet, 0}, {active, false}, {reuseaddr, true}]).
-define(LOGIN_PORT_JP_ONE, 12030). -define(LOGIN_PORT_JP_ONE, 12030).
-define(LOGIN_PORT_JP_TWO, 12031). -define(LOGIN_PORT_JP_TWO, 12031).
-define(LOGIN_PORT_US, 12230). -define(LOGIN_PORT_US, 12230).
-define(LOGIN_LISTEN_OPTIONS, [binary, {active, false}, {certfile, "ssl/servercert.pem"}, {keyfile, "ssl/serverkey.pem"}, {password, "alpha"}]).
-define(GAME_IP, << 188, 165, 56, 119 >>). -define(GAME_IP, << 188, 165, 56, 119 >>).
-define(GAME_PORT, 12061). -define(GAME_PORT, 12061).
-define(GAME_LISTEN_OPTIONS, [binary, {active, false}, {certfile, "ssl/servercert.pem"}, {keyfile, "ssl/serverkey.pem"}, {password, "alpha"}]). -define(GAME_LISTEN_OPTIONS, [binary, {active, false}, {certfile, "priv/ssl/servercert.pem"}, {keyfile, "priv/ssl/serverkey.pem"}, {password, "alpha"}]).

View File

@ -32,7 +32,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). %% @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, { -record(egs_user_model, {
id, pid, socket, state, time, character, instanceid, areatype, area, entryid, pos, id, pid, socket, state, time, character, instancepid, areatype, area, entryid, pos,
%% psu specific fields %% psu specific fields
lid, prev_area, prev_entryid, lid, prev_area, prev_entryid,
%% temporary fields %% temporary fields
@ -98,7 +98,7 @@
%% @doc Table containing all mission objects. %% @doc Table containing all mission objects.
-record(objects, {id, instanceid, objectid, type, targetid, blockid, triggereventid, args}). % id = [instanceid, objectid] -record(psu_object, {id, instancepid, type, args}).
%% @doc Hit response data. %% @doc Hit response data.

View File

@ -70,6 +70,6 @@ db_init() ->
error_logger:info_report("starting mnesia"), error_logger:info_report("starting mnesia"),
mnesia:start(), mnesia:start(),
mnesia:create_table(ids, [{attributes, record_info(fields, ids)}]), mnesia:create_table(ids, [{attributes, record_info(fields, ids)}]),
mnesia:create_table(objects, [{attributes, record_info(fields, objects)}]), mnesia:create_table(psu_object, [{attributes, record_info(fields, psu_object)}]),
mnesia:create_table(egs_user_model, [{attributes, record_info(fields, egs_user_model)}]), mnesia:create_table(egs_user_model, [{attributes, record_info(fields, egs_user_model)}]),
error_logger:info_report("mnesia tables created"). error_logger:info_report("mnesia tables created").

View File

@ -19,56 +19,7 @@
-module(egs_db). -module(egs_db).
-compile(export_all). -compile(export_all).
-include("include/records.hrl").
%% IMPORTANT: The next line must be included
%% if we want to call qlc:q(...)
-include_lib("stdlib/include/qlc.hrl").
do(Q) ->
F = fun() -> qlc:e(Q) end,
{atomic, Val} = mnesia:transaction(F),
Val.
%% @doc Retrieve the next unique ID. %% @doc Retrieve the next unique ID.
%% @todo Used only for the LID so far... %% @todo Used only for the LID so far...
next(Type) -> next(Type) ->
mnesia:dirty_update_counter(ids, Type, 1). mnesia:dirty_update_counter(ids, Type, 1).
%% @doc Insert a new object in the database.
%% @todo Group with users_insert, it's the same function really.
objects_insert(Object) ->
mnesia:transaction(fun() -> mnesia:write(Object) end).
%% @doc Select an object by ID.
objects_select(ID) ->
case mnesia:transaction(fun() -> mnesia:read({objects, ID}) end) of
{atomic, []} ->
undefined;
{atomic, [Val]} ->
Val
end.
%% @doc Select an object by instanceid and targetid.
objects_select_by_targetid(InstanceID, TargetID) ->
[Object] = do(qlc:q([X || X <- mnesia:table(objects),
X#objects.instanceid =:= InstanceID,
X#objects.targetid =:= TargetID])),
Object.
%% @doc Select all IDs.
%% @todo This is for debug purposes only. Delete later.
objects_select_all() ->
do(qlc:q([X || X <- mnesia:table(objects)])).
%% @doc Delete all the objects for the given instance.
objects_delete(InstanceID) ->
List = do(qlc:q([X || X <- mnesia:table(objects), X#objects.instanceid =:= InstanceID])),
[mnesia:transaction(fun() -> mnesia:delete({objects, ID}) end) || #objects{id=ID} <- List],
ok.

View File

@ -39,12 +39,10 @@ do(Q) ->
%% API %% API
%% @spec start_link() -> {ok,Pid::pid()} %% @spec start_link() -> {ok,Pid::pid()}
%% @todo Start the cleanup timer.
start_link() -> start_link() ->
gen_server:start_link({local, ?SERVER}, ?MODULE, [], []). gen_server:start_link({local, ?SERVER}, ?MODULE, [], []).
%% @spec stop() -> stopped %% @spec stop() -> stopped
%% @todo Stop the cleanup timer.
stop() -> stop() ->
gen_server:call(?SERVER, stop). gen_server:call(?SERVER, stop).
@ -103,7 +101,7 @@ handle_call({select, {neighbors, User}}, _From, State) ->
List = do(qlc:q([X || X <- mnesia:table(?TABLE), List = do(qlc:q([X || X <- mnesia:table(?TABLE),
X#?TABLE.id /= User#?TABLE.id, X#?TABLE.id /= User#?TABLE.id,
X#?TABLE.state =:= online, X#?TABLE.state =:= online,
X#?TABLE.instanceid =:= User#?TABLE.instanceid, X#?TABLE.instancepid =:= User#?TABLE.instancepid,
X#?TABLE.area =:= User#?TABLE.area X#?TABLE.area =:= User#?TABLE.area
])), ])),
{reply, {ok, List}, State}; {reply, {ok, List}, State};

View File

@ -23,6 +23,7 @@
-include("include/records.hrl"). -include("include/records.hrl").
-include("include/maps.hrl"). -include("include/maps.hrl").
-include("include/missions.hrl").
-define(OPTIONS, [binary, {active, false}, {certfile, "priv/ssl/servercert.pem"}, {keyfile, "priv/ssl/serverkey.pem"}, {password, "alpha"}]). -define(OPTIONS, [binary, {active, false}, {certfile, "priv/ssl/servercert.pem"}, {keyfile, "priv/ssl/serverkey.pem"}, {password, "alpha"}]).
@ -35,6 +36,7 @@ start_link(Port) ->
%% @spec cleanup(Pid) -> ok %% @spec cleanup(Pid) -> ok
%% @doc Cleanup the data associated with the failing process. %% @doc Cleanup the data associated with the failing process.
%% @todo Cleanup the instance process if there's nobody in it anymore.
cleanup(Pid) -> cleanup(Pid) ->
case egs_user_model:read({pid, Pid}) of case egs_user_model:read({pid, Pid}) of
{ok, User} -> {ok, User} ->
@ -314,20 +316,20 @@ area_get_season(QuestID) ->
area_load(QuestID, ZoneID, MapID, EntryID) -> area_load(QuestID, ZoneID, MapID, EntryID) ->
{ok, OldUser} = egs_user_model:read(get(gid)), {ok, OldUser} = egs_user_model:read(get(gid)),
[{type, AreaType}, {file, QuestFile}|StartInfo] = proplists:get_value(QuestID, ?QUESTS, [{type, undefined}, {file, undefined}]), [{type, AreaType}, {file, QuestFile}|StartInfo] = proplists:get_value(QuestID, ?QUESTS, [{type, undefined}, {file, undefined}]),
[IsStart, InstanceID, RealZoneID, RealMapID, RealEntryID] = case AreaType of [IsStart, RealZoneID, RealMapID, RealEntryID] = case AreaType of
mission -> mission ->
if ZoneID =:= 65535 -> if ZoneID =:= 65535 ->
[{start, [TmpZoneID, TmpMapID, TmpEntryID]}] = StartInfo, [{start, [TmpZoneID, TmpMapID, TmpEntryID]}] = StartInfo,
[true, OldUser#egs_user_model.id, TmpZoneID, TmpMapID, TmpEntryID]; [true, TmpZoneID, TmpMapID, TmpEntryID];
true -> [false, OldUser#egs_user_model.id, ZoneID, MapID, EntryID] true -> [false, ZoneID, MapID, EntryID]
end; end;
myroom -> myroom ->
if ZoneID =:= 0 -> if ZoneID =:= 0 ->
[false, undefined, 0, 423, EntryID]; [false, 0, 423, EntryID];
true -> [false, undefined, ZoneID, MapID, EntryID] true -> [false, ZoneID, MapID, EntryID]
end; end;
_ -> _ ->
[false, undefined, ZoneID, MapID, EntryID] [false, ZoneID, MapID, EntryID]
end, end,
[{file, ZoneFile}] = proplists:get_value([QuestID, RealZoneID], ?ZONES, [{file, undefined}]), [{file, ZoneFile}] = proplists:get_value([QuestID, RealZoneID], ?ZONES, [{file, undefined}]),
if AreaType =:= myroom -> if AreaType =:= myroom ->
@ -335,14 +337,16 @@ area_load(QuestID, ZoneID, MapID, EntryID) ->
true -> true ->
[{name, AreaName}] = proplists:get_value([QuestID, RealMapID], ?MAPS, [{name, "dammy"}]) [{name, AreaName}] = proplists:get_value([QuestID, RealMapID], ?MAPS, [{name, "dammy"}])
end, end,
User = OldUser#egs_user_model{instanceid=InstanceID, areatype=AreaType, area={psu_area, QuestID, RealZoneID, RealMapID}, entryid=RealEntryID},
egs_user_model:write(User),
%% @todo Don't recalculate SetID when leaving the mission and back (this changes the spawns). %% @todo Don't recalculate SetID when leaving the mission and back (this changes the spawns).
SetID = if IsStart =:= true -> crypto:rand_uniform(0, 4); true -> 0 end, SetID = if IsStart =:= true -> crypto:rand_uniform(0, 4); true -> 0 end,
if IsStart =:= true -> % initialize the mission InstancePid = if IsStart =:= true -> % initialize the mission
psu_missions:start(InstanceID, QuestID, SetID); Zones = proplists:get_value(QuestID, ?MISSIONS),
true -> ignore {ok, RetPid} = psu_instance:start_link(Zones),
RetPid;
true -> OldUser#egs_user_model.instancepid
end, end,
User = OldUser#egs_user_model{instancepid=InstancePid, areatype=AreaType, area={psu_area, QuestID, RealZoneID, RealMapID}, entryid=RealEntryID},
egs_user_model:write(User),
area_load(AreaType, IsStart, SetID, OldUser, User, QuestFile, ZoneFile, AreaName). area_load(AreaType, IsStart, SetID, OldUser, User, QuestFile, ZoneFile, AreaName).
area_load(AreaType, IsStart, SetID, OldUser, User, QuestFile, ZoneFile, AreaName) -> area_load(AreaType, IsStart, SetID, OldUser, User, QuestFile, ZoneFile, AreaName) ->
@ -702,7 +706,7 @@ handle(16#0402, Data) ->
7 -> % spawn cleared @todo 1201 sent back with same values apparently, but not always 7 -> % spawn cleared @todo 1201 sent back with same values apparently, but not always
log("cleared spawn ~b", [SpawnID]), log("cleared spawn ~b", [SpawnID]),
{ok, User} = egs_user_model:read(get(gid)), {ok, User} = egs_user_model:read(get(gid)),
[EventID, BlockID] = psu_missions:spawn_cleared(User#egs_user_model.instanceid, SpawnID), {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; if EventID =:= false -> ignore;
true -> send_1205(EventID, BlockID, 0) true -> send_1205(EventID, BlockID, 0)
end; end;
@ -781,7 +785,7 @@ handle(16#0c0e, _) ->
send_1006(11), send_1006(11),
{ok, User} = egs_user_model:read(get(gid)), {ok, User} = egs_user_model:read(get(gid)),
%% delete the mission %% delete the mission
psu_missions:stop(User#egs_user_model.instanceid), psu_instance:stop(User#egs_user_model.instancepid),
%% full hp %% full hp
Character = User#egs_user_model.character, Character = User#egs_user_model.character,
MaxHP = Character#characters.maxhp, MaxHP = Character#characters.maxhp,
@ -843,8 +847,8 @@ handle(16#0f0a, Data) ->
case Action of case Action of
0 -> % warp 0 -> % warp
{ok, User} = egs_user_model:read(get(gid)), {ok, User} = egs_user_model:read(get(gid)),
{X, Y, Z, Dir} = psu_missions:warp_event(User#egs_user_model.instanceid, BlockID, ListNb, ObjectNb), 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{x=X, y=Y, z=Z, dir=Dir}}, NewUser = User#egs_user_model{pos=Pos},
egs_user_model:write(NewUser), egs_user_model:write(NewUser),
send_0503(User#egs_user_model.pos), send_0503(User#egs_user_model.pos),
send_1211(A, C, B, 0); send_1211(A, C, B, 0);
@ -866,7 +870,7 @@ handle(16#0f0a, Data) ->
ignore; ignore;
12 -> % pick/use key 12 -> % pick/use key
{ok, User} = egs_user_model:read(get(gid)), {ok, User} = egs_user_model:read(get(gid)),
[[EventID|_], BlockID] = psu_missions:key_event(User#egs_user_model.instanceid, ObjectID), {BlockID, [EventID|_]} = psu_instance:key_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, ObjectID),
send_1205(EventID, BlockID, 0), send_1205(EventID, BlockID, 0),
send_1213(ObjectID, 1); send_1213(ObjectID, 1);
13 -> % floor_button on (also sent when clearing a few of the rooms in black nest) 13 -> % floor_button on (also sent when clearing a few of the rooms in black nest)
@ -881,11 +885,11 @@ handle(16#0f0a, Data) ->
ignore; ignore;
23 -> % initialize key slots (called when picking a key or checking the gate directly with no key) 23 -> % initialize key slots (called when picking a key or checking the gate directly with no key)
{ok, User} = egs_user_model:read(get(gid)), {ok, User} = egs_user_model:read(get(gid)),
[[_, EventID, _], BlockID] = psu_missions:key_event(User#egs_user_model.instanceid, ObjectID), {BlockID, [_, EventID, _]} = psu_instance:key_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, ObjectID),
send_1205(EventID, BlockID, 0); % in block 1, 202 = key [1] x1, 203 = key [-] x1 send_1205(EventID, BlockID, 0); % in block 1, 202 = key [1] x1, 203 = key [-] x1
24 -> % open gate (only when client has key) 24 -> % open gate (only when client has key)
{ok, User} = egs_user_model:read(get(gid)), {ok, User} = egs_user_model:read(get(gid)),
[[_, _, EventID], BlockID] = psu_missions:key_event(User#egs_user_model.instanceid, ObjectID), {BlockID, [_, _, EventID]} = psu_instance:key_event(User#egs_user_model.instancepid, (User#egs_user_model.area)#psu_area.zoneid, ObjectID),
send_1205(EventID, BlockID, 0), send_1205(EventID, BlockID, 0),
send_1213(ObjectID, 1); send_1213(ObjectID, 1);
25 -> % sit on chair 25 -> % sit on chair
@ -1006,7 +1010,7 @@ handle_hits(Data) ->
GID = get(gid), GID = get(gid),
{ok, User} = egs_user_model:read(GID), {ok, User} = egs_user_model:read(GID),
% hit! % hit!
#hit_response{type=Type, user=NewUser, exp=HasEXP, damage=Damage, targethp=TargetHP, targetse=TargetSE, events=Events} = psu_missions:object_hit(User, SourceID, TargetID), #hit_response{type=Type, user=NewUser, exp=HasEXP, damage=Damage, targethp=TargetHP, targetse=TargetSE, events=Events} = psu_instance:hit(User, SourceID, TargetID),
case Type of case Type of
box -> box ->
% TODO: also has a hit sent, we should send it too % TODO: also has a hit sent, we should send it too
@ -1042,7 +1046,7 @@ handle_events([]) ->
handle_events([{explode, ObjectID}|Tail]) -> handle_events([{explode, ObjectID}|Tail]) ->
send_1213(ObjectID, 3), send_1213(ObjectID, 3),
handle_events(Tail); handle_events(Tail);
handle_events([{event, [EventID, BlockID]}|Tail]) -> handle_events([{event, [BlockID, EventID]}|Tail]) ->
send_1205(EventID, BlockID, 0), send_1205(EventID, BlockID, 0),
handle_events(Tail). handle_events(Tail).

239
src/psu/psu_instance.erl Normal file
View File

@ -0,0 +1,239 @@
%% @author Loïc Hoguin <essen@dev-extend.eu>
%% @copyright 2010 Loïc Hoguin.
%% @doc Handle instances, their objects and their events.
%%
%% 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 <http://www.gnu.org/licenses/>.
-module(psu_instance).
-behavior(gen_server).
-export([start_link/1, stop/1, key_event/3, spawn_cleared_event/3, warp_event/5, hit/3]). %% API.
-export([init/1, handle_call/3, handle_cast/2, handle_info/2, terminate/2, code_change/3]). %% gen_server.
-include_lib("stdlib/include/qlc.hrl").
-include("include/records.hrl").
%% @spec do(Q) -> Record
%% @doc Perform a mnesia transaction using a QLC query.
do(Q) ->
F = fun() -> qlc:e(Q) end,
{atomic, Val} = mnesia:transaction(F),
Val.
%% API.
%% @spec start_link() -> {ok,Pid::pid()}
start_link(Zones) ->
gen_server:start_link(?MODULE, [Zones], []).
%% @spec stop() -> stopped
stop(InstancePid) ->
gen_server:call(InstancePid, stop).
%% @todo @spec event(ServerPid, ObjectID, Args) -> Response
key_event(InstancePid, ZoneID, ObjectID) ->
gen_server:call(InstancePid, {key_event, ZoneID, ObjectID}).
spawn_cleared_event(InstancePid, ZoneID, SpawnID) ->
gen_server:call(InstancePid, {spawn_cleared_event, ZoneID, SpawnID}).
warp_event(InstancePid, ZoneID, BlockID, ListIndex, ObjectIndex) ->
gen_server:call(InstancePid, {warp_event, ZoneID, BlockID, ListIndex, ObjectIndex}).
%% @todo @spec hit(ServerPid, TargetID, Args) -> Response
hit(User, SourceID, TargetID) ->
gen_server:call(User#egs_user_model.instancepid, {hit, (User#egs_user_model.area)#psu_area.zoneid, User, SourceID, TargetID}).
%% gen_server.
init([Zones]) ->
[map_init(Maps, ZoneID, 0, 0, 1024) || {ZoneID, Maps} <- Zones],
{ok, undefined}.
map_init(undefined, _ZoneID, _BlockID, _ObjectID, _TargetID) ->
ignore; %% @todo Temporary clause until everything works fine.
map_init([], _ZoneID, _BlockID, _ObjectID, _TargetID) ->
ok;
map_init([{_MapID, ObjectLists}|Tail], ZoneID, BlockID, ObjectID, TargetID) ->
{ok, NextObjectID, NextTargetID} = list_init(ObjectLists, ZoneID, BlockID, ObjectID, TargetID, 0),
map_init(Tail, ZoneID, BlockID + 1, NextObjectID, NextTargetID).
list_init([], _ZoneID, _BlockID, ObjectID, TargetID, _ListIndex) ->
{ok, ObjectID, TargetID};
list_init([Objects|Tail], ZoneID, BlockID, ObjectID, TargetID, ListIndex) ->
{ok, NextObjectID, NextTargetID} = object_init(Objects, ZoneID, BlockID, ObjectID, TargetID, ListIndex, 0),
list_init(Tail, ZoneID, BlockID, NextObjectID, NextTargetID, ListIndex + 1).
object_init([], _ZoneID, _BlockID, ObjectID, TargetID, _ListIndex, _ObjectIndex) ->
{ok, ObjectID, TargetID};
%% @doc box: {InstancePid, ZoneID, TargetID}
object_init([{box, _Model, Breakable, TrigEventID}|Tail], ZoneID, BlockID, ObjectID, TargetID, ListIndex, ObjectIndex) ->
case Breakable of
false -> ignore;
true -> object_insert(#psu_object{id={self(), ZoneID, TargetID}, instancepid=self(), type=box, args={BlockID, ObjectID, TrigEventID}})
end,
object_init(Tail, ZoneID, BlockID, ObjectID + 1, TargetID + 1, ListIndex, ObjectIndex + 1);
%% @doc key: {InstancePid, ZoneID, key, ObjectID}
object_init([{key, _KeySet, TrigEventID, _ReqEventID}|Tail], ZoneID, BlockID, ObjectID, TargetID, ListIndex, ObjectIndex) ->
object_insert(#psu_object{id={self(), ZoneID, key, ObjectID}, instancepid=self(), type=key, args={BlockID, [TrigEventID]}}),
object_init(Tail, ZoneID, BlockID, ObjectID + 1, TargetID, ListIndex, ObjectIndex + 1);
%% @doc key_console: @see key; @todo handled the same for now
object_init([{key_console, KeySet, TrigEventID, _ReqEventID}|Tail], ZoneID, BlockID, ObjectID, TargetID, ListIndex, ObjectIndex) ->
object_insert(#psu_object{id={self(), ZoneID, key, ObjectID}, instancepid=self(), type=key_console, args={BlockID, [243 + KeySet, 201 + KeySet, TrigEventID]}}),
object_init(Tail, ZoneID, BlockID, ObjectID + 1, TargetID, ListIndex, ObjectIndex + 1);
%% @doc spawn: {InstancePid, ZoneID, 'spawn', SpawnID}
%% @todo save enemies individually, do something, etc.
%% @todo temporarily save the spawn to handle events properly
object_init([{'spawn', NbTargets, TrigEventID, _ReqEventID}|Tail], ZoneID, BlockID, ObjectID, TargetID, ListIndex, ObjectIndex) ->
object_insert(#psu_object{id={self(), ZoneID, 'spawn', TargetID - 1024}, instancepid=self(), type='spawn', args={BlockID, TrigEventID}}),
object_init(Tail, ZoneID, BlockID, ObjectID + 1, TargetID + NbTargets, ListIndex, ObjectIndex + 1);
%% @doc warp: {InstancePid, ZoneID, warp, BlockID, ListIndex, ObjectIndex}
object_init([{warp, DestX, DestY, DestZ, DestDir}|Tail], ZoneID, BlockID, ObjectID, TargetID, ListIndex, ObjectIndex) ->
object_insert(#psu_object{id={self(), ZoneID, warp, BlockID, ListIndex, ObjectIndex}, instancepid=self(), type=warp, args=#pos{x=DestX, y=DestY, z=DestZ, dir=DestDir}}),
object_init(Tail, ZoneID, BlockID, ObjectID, TargetID, ListIndex, ObjectIndex + 1);
%% @doc Ignore for now: crystal
%% @todo Probably have 2 TargetID because of its on/off state.
object_init([crystal|Tail], ZoneID, BlockID, ObjectID, TargetID, ListIndex, ObjectIndex) ->
object_init(Tail, ZoneID, BlockID, ObjectID + 1, TargetID + 2, ListIndex, ObjectIndex + 1);
%% @doc Ignore for now: floor_button, shoot_button, google_target, trap (all kinds)
object_init([Object|Tail], ZoneID, BlockID, ObjectID, TargetID, ListIndex, ObjectIndex)
when Object =:= floor_button;
Object =:= shoot_button;
Object =:= google_target;
Object =:= trap ->
object_init(Tail, ZoneID, BlockID, ObjectID + 1, TargetID + 1, ListIndex, ObjectIndex + 1);
%% @doc Ignore for now: objects without any ObjectID or TargetID.
object_init([Object|Tail], ZoneID, BlockID, ObjectID, TargetID, ListIndex, ObjectIndex)
when Object =:= static_model;
Object =:= invisible_block;
Object =:= entrance;
Object =:= 'exit';
Object =:= label;
Object =:= hidden_minimap_section;
Object =:= fog;
Object =:= pp_cube;
Object =:= healing_pad ->
object_init(Tail, ZoneID, BlockID, ObjectID, TargetID, ListIndex, ObjectIndex + 1);
%% @doc Ignore everything else for now: objects with an ObjectID but without a TargetID.
object_init([_Object|Tail], ZoneID, BlockID, ObjectID, TargetID, ListIndex, ObjectIndex) ->
object_init(Tail, ZoneID, BlockID, ObjectID + 1, TargetID, ListIndex, ObjectIndex + 1).
%% Event handlers
handle_call({key_event, ZoneID, ObjectID}, _From, State) ->
#psu_object{args=Args} = object_select({self(), ZoneID, key, ObjectID}),
{reply, Args, State};
handle_call({spawn_cleared_event, ZoneID, SpawnID}, _From, State) ->
#psu_object{args=Args} = object_select({self(), ZoneID, 'spawn', SpawnID}),
{reply, Args, State};
handle_call({warp_event, ZoneID, BlockID, ListIndex, ObjectIndex}, _From, State) ->
#psu_object{args=Args} = object_select({self(), ZoneID, warp, BlockID, ListIndex, ObjectIndex}),
{reply, Args, State};
%% Hit handler
%% @todo Handle everything correctly.
handle_call({hit, ZoneID, User, _SourceID, TargetID}, _From, State) ->
try
Box = object_select({self(), ZoneID, TargetID}),
BoxReply = box_hit(Box),
{reply, BoxReply, State}
catch _:_ ->
OtherReply = if TargetID =:= 0 ->
player_hit(User);
true -> enemy_hit(User)
end,
{reply, OtherReply, State}
end;
%% @doc Remove all objects from the database when the process is stopped.
handle_call(stop, _From, State) ->
List = do(qlc:q([X || X <- mnesia:table(psu_object), X#psu_object.instancepid =:= self()])),
[mnesia:transaction(fun() -> mnesia:delete({psu_object, ID}) end) || #psu_object{id=ID} <- List],
{stop, normal, stopped, State};
handle_call(_Request, _From, State) ->
{reply, ignored, State}.
handle_cast(_Msg, State) ->
{noreply, State}.
handle_info(_Info, State) ->
{noreply, State}.
terminate(_Reason, _State) ->
ok.
code_change(_OldVsn, State, _Extra) ->
{ok, State}.
%% Internal functions
%% @todo doc
object_insert(Object) ->
mnesia:transaction(fun() -> mnesia:write(Object) end).
%% @todo doc
object_select(ID) ->
case mnesia:transaction(fun() -> mnesia:read({psu_object, ID}) end) of
{atomic, []} -> undefined;
{atomic, [Val]} -> Val
end.
box_hit(#psu_object{args={BlockID, ObjectID, TrigEventID}}) ->
Events =
if TrigEventID =:= false -> [{explode, ObjectID}];
true -> [{explode, ObjectID}, {event, [BlockID, TrigEventID]}]
end,
#hit_response{type=box, events=Events}.
enemy_hit(User) ->
Damage = 1,
IncEXP = 1,
Character = User#egs_user_model.character,
Level = Character#characters.mainlevel,
NewEXP = Level#level.exp + IncEXP,
NewLevel = Level#level{exp=NewEXP},
NewCharacter = Character#characters{mainlevel=NewLevel},
NewUser = User#egs_user_model{character=NewCharacter},
% todo delete the enemy from the db when it dies
#hit_response{type=enemy, user=NewUser, exp=true, damage=Damage, targethp=0, targetse=[death]}.
player_hit(User) ->
Damage = 10,
Character = User#egs_user_model.character,
TmpHP = Character#characters.currenthp - Damage,
if TmpHP =< 0 ->
NewHP = 0,
SE = [flinch, death];
true ->
NewHP = TmpHP,
SE = [flinch]
end,
NewCharacter = Character#characters{currenthp=NewHP},
NewUser = User#egs_user_model{character=NewCharacter},
#hit_response{type=player, user=NewUser, damage=Damage, targethp=NewHP, targetse=SE}.

View File

@ -1,169 +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 <http://www.gnu.org/licenses/>.
-module(psu_missions).
-export([
start/3, stop/1, key_event/2, warp_event/4, object_hit/3, spawn_cleared/2
]).
-include("include/missions.hrl").
-include("include/records.hrl").
start(InstanceID, QuestID, _SetID) ->
MapList = proplists:get_value(QuestID, ?MISSIONS),
map_init(InstanceID, MapList, 0, 0, 1024).
map_init(_InstanceID, undefined, _BlockID, _ObjectID, _TargetID) ->
ignore; %% @todo Temporary clause until everything works fine.
map_init(_InstanceID, [], _BlockID, _ObjectID, _TargetID) ->
ok;
map_init(InstanceID, [Map|Tail], BlockID, ObjectID, TargetID) ->
{_MapID, Objects} = Map,
{ok, NewObjectID, NewTargetID} = list_init(InstanceID, BlockID, Objects, 0, ObjectID, TargetID),
map_init(InstanceID, Tail, BlockID + 1, NewObjectID, NewTargetID).
list_init(_InstanceID, _BlockID, [], _ListNb, ObjectID, TargetID) ->
{ok, ObjectID, TargetID};
list_init(InstanceID, BlockID, [Objects|Tail], ListNb, ObjectID, TargetID) ->
{ok, NewObjectID, NewTargetID} = object_init(InstanceID, BlockID, Objects, ListNb, 0, ObjectID, TargetID),
list_init(InstanceID, BlockID, Tail, ListNb + 1, NewObjectID, NewTargetID).
object_init(_InstanceID, _BlockID, [], _ListNb, _ObjectNb, ObjectID, TargetID) ->
{ok, ObjectID, TargetID};
object_init(InstanceID, BlockID, [{box, _Model, Breakable, TrigEventID}|Tail], ListNb, ObjectNb, ObjectID, TargetID) ->
case Breakable of
false -> ignore;
true ->
egs_db:objects_insert(#objects{id=[InstanceID, ObjectID], instanceid=InstanceID, objectid=ObjectID, type=box, targetid=TargetID, blockid=BlockID, triggereventid=TrigEventID})
end,
object_init(InstanceID, BlockID, Tail, ListNb, ObjectNb + 1, ObjectID + 1, TargetID + 1);
%% @todo Apparently floor_button has a TargetID. Not sure why yet. Just increment the value.
object_init(InstanceID, BlockID, [floor_button|Tail], ListNb, ObjectNb, ObjectID, TargetID) ->
object_init(InstanceID, BlockID, Tail, ListNb, ObjectNb + 1, ObjectID + 1, TargetID + 1);
%% @todo Apparently shoot_button has a TargetID. I'm sure why though.
object_init(InstanceID, BlockID, [shoot_button|Tail], ListNb, ObjectNb, ObjectID, TargetID) ->
object_init(InstanceID, BlockID, Tail, ListNb, ObjectNb + 1, ObjectID + 1, TargetID + 1);
%% @todo Goggle targets have a targetid.
object_init(InstanceID, BlockID, [goggle_target|Tail], ListNb, ObjectNb, ObjectID, TargetID) ->
object_init(InstanceID, BlockID, Tail, ListNb, ObjectNb + 1, ObjectID + 1, TargetID + 1);
%% @todo All kinds of traps have a TargetID, even if they're not targettable.
object_init(InstanceID, BlockID, [trap|Tail], ListNb, ObjectNb, ObjectID, TargetID) ->
object_init(InstanceID, BlockID, Tail, ListNb, ObjectNb + 1, ObjectID + 1, TargetID + 1);
%% @todo Not sure why but it works that way in True Darkness.
object_init(InstanceID, BlockID, [boss_gate|Tail], ListNb, ObjectNb, ObjectID, TargetID) ->
object_init(InstanceID, BlockID, Tail, ListNb, ObjectNb + 1, ObjectID + 1, TargetID + 1);
%% @todo key and key_console event handling will have to be fixed.
object_init(InstanceID, BlockID, [{key, _KeySet, TrigEventID, _ReqEventID}|Tail], ListNb, ObjectNb, ObjectID, TargetID) ->
egs_db:objects_insert(#objects{id=[InstanceID, {key, ObjectID}], instanceid=InstanceID, objectid=ObjectID, type=key, blockid=BlockID, triggereventid=[TrigEventID]}),
object_init(InstanceID, BlockID, Tail, ListNb, ObjectNb + 1, ObjectID + 1, TargetID);
%% @todo Maybe separate key from key_console in its handling?
object_init(InstanceID, BlockID, [{key_console, KeySet, _ReqKeyEventsID, TrigEventID}|Tail], ListNb, ObjectNb, ObjectID, TargetID) ->
egs_db:objects_insert(#objects{id=[InstanceID, {key, ObjectID}], instanceid=InstanceID, objectid=ObjectID, type=key, blockid=BlockID, triggereventid=[243 + KeySet, 201 + KeySet, TrigEventID]}),
object_init(InstanceID, BlockID, Tail, ListNb, ObjectNb + 1, ObjectID + 1, TargetID);
%% @todo save enemies individually, do something, etc.
%% @todo temporarily save the spawn to handle events properly
object_init(InstanceID, BlockID, [{'spawn', NbTargets, TrigEventID, _ReqEventID}|Tail], ListNb, ObjectNb, ObjectID, TargetID) ->
egs_db:objects_insert(#objects{id=[InstanceID, {'spawn', TargetID - 1024}], instanceid=InstanceID, type='spawn', blockid=BlockID, triggereventid=TrigEventID}),
object_init(InstanceID, BlockID, Tail, ListNb, ObjectNb + 1, ObjectID + 1, TargetID + NbTargets);
object_init(InstanceID, BlockID, [{warp, DestX, DestY, DestZ, DestDir}|Tail], ListNb, ObjectNb, ObjectID, TargetID) ->
egs_db:objects_insert(#objects{id=[InstanceID, {warp, BlockID, ListNb, ObjectNb}], instanceid=InstanceID, type=warp, blockid=BlockID, args={DestX, DestY, DestZ, DestDir}}),
object_init(InstanceID, BlockID, Tail, ListNb, ObjectNb + 1, ObjectID, TargetID);
%% @todo Apparently crystal has 2 TargetIDs. Presumably because it has both on/off states.
object_init(InstanceID, BlockID, [crystal|Tail], ListNb, ObjectNb, ObjectID, TargetID) ->
object_init(InstanceID, BlockID, Tail, ListNb, ObjectNb + 1, ObjectID + 1, TargetID + 2);
%% A few object types don't have an ObjectID nor a TargetID. Disregard them completely.
object_init(InstanceID, BlockID, [ObjType|Tail], ListNb, ObjectNb, ObjectID, TargetID)
when ObjType =:= static_model;
ObjType =:= invisible_block;
ObjType =:= entrance;
ObjType =:= 'exit';
ObjType =:= label;
ObjType =:= hidden_minimap_section;
ObjType =:= fog;
ObjType =:= pp_cube;
ObjType =:= healing_pad ->
object_init(InstanceID, BlockID, Tail, ListNb, ObjectNb + 1, ObjectID, TargetID);
%% Others are normal objects, we don't handle them but they have an ObjectID.
object_init(InstanceID, BlockID, [_|Tail], ListNb, ObjectNb, ObjectID, TargetID) ->
object_init(InstanceID, BlockID, Tail, ListNb, ObjectNb + 1, ObjectID + 1, TargetID).
stop(InstanceID) ->
egs_db:objects_delete(InstanceID).
key_event(InstanceID, ObjectID) ->
#objects{triggereventid=EventID, blockid=BlockID} = egs_db:objects_select([InstanceID, {key, ObjectID}]),
[EventID, BlockID].
warp_event(InstanceID, BlockID, ListNb, ObjectNb) ->
#objects{args=Args} = egs_db:objects_select([InstanceID, {warp, BlockID, ListNb, ObjectNb}]),
Args.
object_hit(User, _SourceID, TargetID) ->
try
Object = egs_db:objects_select_by_targetid(User#egs_user_model.instanceid, TargetID),
if Object#objects.type =:= box ->
box_hit(User, Object);
true ->
io:format("unknown object hit~n")
end
catch _:_ ->
if TargetID =:= 0 ->
player_hit(User);
true ->
enemy_hit(User)
end
end.
box_hit(User, Box) ->
% todo delete the box from the db
EventsResponse =
if Box#objects.triggereventid =:= false -> [{explode, Box#objects.objectid}];
true -> [{explode, Box#objects.objectid}, {event, [Box#objects.triggereventid, Box#objects.blockid]}]
end,
#hit_response{type=box, user=User, events=EventsResponse}.
enemy_hit(User) ->
Damage = 1,
IncEXP = 1,
Character = User#egs_user_model.character,
Level = Character#characters.mainlevel,
NewEXP = Level#level.exp + IncEXP,
NewLevel = Level#level{exp=NewEXP},
NewCharacter = Character#characters{mainlevel=NewLevel},
NewUser = User#egs_user_model{character=NewCharacter},
% todo delete the enemy from the db when it dies
#hit_response{type=enemy, user=NewUser, exp=true, damage=Damage, targethp=0, targetse=[death]}.
player_hit(User) ->
Damage = 10,
Character = User#egs_user_model.character,
TmpHP = Character#characters.currenthp - Damage,
if TmpHP =< 0 ->
NewHP = 0,
SE = [flinch, death];
true ->
NewHP = TmpHP,
SE = [flinch]
end,
NewCharacter = Character#characters{currenthp=NewHP},
NewUser = User#egs_user_model{character=NewCharacter},
#hit_response{type=player, user=NewUser, damage=Damage, targethp=NewHP, targetse=SE}.
spawn_cleared(InstanceID, SpawnID) ->
#objects{triggereventid=EventID, blockid=BlockID} = egs_db:objects_select([InstanceID, {'spawn', SpawnID}]),
[EventID, BlockID].

View File

@ -45,7 +45,7 @@ parse_zone(QuestID, NblFilename) ->
log("header: end ptr(~b) areaid list ptr(~b)", [EndRelPtr, AreaIDListRelPtr]), log("header: end ptr(~b) areaid list ptr(~b)", [EndRelPtr, AreaIDListRelPtr]),
{ok, _AreaCode, NbMaps, MapsListPtr} = parse_areaid_list(Data, AreaIDListRelPtr - 16), {ok, _AreaCode, NbMaps, MapsListPtr} = parse_areaid_list(Data, AreaIDListRelPtr - 16),
MapList = parse_mapnumbers_list(Data, NbMaps, MapsListPtr - BasePtr - 16), MapList = parse_mapnumbers_list(Data, NbMaps, MapsListPtr - BasePtr - 16),
ObjList = {QuestID, [{MapID, parse_object_list_headers(BasePtr, Data, NbHeaders, ObjListHeadersPtr - BasePtr - 16)} || {MapID, NbHeaders, ObjListHeadersPtr} <- MapList]}, ObjList = {QuestID, [{0, [{MapID, parse_object_list_headers(BasePtr, Data, NbHeaders, ObjListHeadersPtr - BasePtr - 16)} || {MapID, NbHeaders, ObjListHeadersPtr} <- MapList]}]},
nbl_cleanup(), nbl_cleanup(),
ObjList. ObjList.
@ -172,7 +172,7 @@ parse_object_args(22, _Params, Data) ->
ReqKeyEvents = [convert_eventid(RawReqKey1Event), convert_eventid(RawReqKey2Event), convert_eventid(RawReqKey3Event), convert_eventid(RawReqKey4Event)], ReqKeyEvents = [convert_eventid(RawReqKey1Event), convert_eventid(RawReqKey2Event), convert_eventid(RawReqKey3Event), convert_eventid(RawReqKey4Event)],
TrigEvent = convert_eventid(RawTrigEvent), TrigEvent = convert_eventid(RawTrigEvent),
log("key_console: a(~b) keyset(~b) b(~b) c(~b) reqkeyevents(~p) trigevent(~p) d(~b)", [UnknownA, KeySet, UnknownB, UnknownC, ReqKeyEvents, TrigEvent, UnknownD]), log("key_console: a(~b) keyset(~b) b(~b) c(~b) reqkeyevents(~p) trigevent(~p) d(~b)", [UnknownA, KeySet, UnknownB, UnknownC, ReqKeyEvents, TrigEvent, UnknownD]),
{key_console, KeySet, ReqKeyEvents, TrigEvent}; {key_console, KeySet, TrigEvent, ReqKeyEvents};
%% @doc Small spawn. %% @doc Small spawn.