2010-07-19 08:11:31 +08:00
|
|
|
|
%% @author Lo<4C>c Hoguin <essen@dev-extend.eu>
|
|
|
|
|
%% @copyright 2010 Lo<4C>c Hoguin.
|
2011-02-13 00:53:58 +08:00
|
|
|
|
%% @doc Users handling.
|
2010-07-19 08:11:31 +08:00
|
|
|
|
%%
|
|
|
|
|
%% This file is part of EGS.
|
|
|
|
|
%%
|
|
|
|
|
%% EGS is free software: you can redistribute it and/or modify
|
2010-09-04 06:09:06 +08:00
|
|
|
|
%% it under the terms of the GNU Affero General Public License as
|
|
|
|
|
%% published by the Free Software Foundation, either version 3 of the
|
|
|
|
|
%% License, or (at your option) any later version.
|
2010-07-19 08:11:31 +08:00
|
|
|
|
%%
|
|
|
|
|
%% 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
|
2010-09-04 06:09:06 +08:00
|
|
|
|
%% GNU Affero General Public License for more details.
|
2010-07-19 08:11:31 +08:00
|
|
|
|
%%
|
2010-09-04 06:09:06 +08:00
|
|
|
|
%% You should have received a copy of the GNU Affero General Public License
|
2010-07-19 08:11:31 +08:00
|
|
|
|
%% along with EGS. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
2011-02-13 00:53:58 +08:00
|
|
|
|
-module(egs_users).
|
2011-02-20 02:45:50 +08:00
|
|
|
|
-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]).
|
2010-07-19 08:11:31 +08:00
|
|
|
|
|
2011-02-13 00:53:58 +08:00
|
|
|
|
-define(TABLE, users).
|
2010-07-19 08:11:31 +08:00
|
|
|
|
|
|
|
|
|
-include("include/records.hrl").
|
|
|
|
|
-include_lib("stdlib/include/qlc.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.
|
|
|
|
|
|
2011-02-13 00:53:58 +08:00
|
|
|
|
%% --
|
2010-07-19 08:11:31 +08:00
|
|
|
|
|
|
|
|
|
%% @spec read({pid, Pid}) -> {ok, User} | {error, badarg}
|
|
|
|
|
%% @spec read(ID) -> {ok, User} | {error, badarg}
|
2011-02-13 00:53:58 +08:00
|
|
|
|
read({pid, Pid}) ->
|
2010-07-19 08:11:31 +08:00
|
|
|
|
List = do(qlc:q([X || X <- mnesia:table(?TABLE), X#?TABLE.pid =:= Pid])),
|
|
|
|
|
case List of
|
2011-02-13 00:53:58 +08:00
|
|
|
|
[] -> {error, badarg};
|
|
|
|
|
[User] -> {ok, User}
|
2010-07-19 08:11:31 +08:00
|
|
|
|
end;
|
2011-02-13 00:53:58 +08:00
|
|
|
|
read(ID) ->
|
2010-07-19 08:11:31 +08:00
|
|
|
|
case mnesia:transaction(fun() -> mnesia:read({?TABLE, ID}) end) of
|
2011-02-13 00:53:58 +08:00
|
|
|
|
{atomic, []} -> {error, badarg};
|
|
|
|
|
{atomic, [Val]} -> {ok, Val}
|
|
|
|
|
end.
|
2010-07-19 08:11:31 +08:00
|
|
|
|
|
2011-02-13 00:53:58 +08:00
|
|
|
|
%% @spec select(all) -> {ok, List}
|
|
|
|
|
%% @spec select({neighbors, User}) -> {ok, List}
|
2010-07-19 08:11:31 +08:00
|
|
|
|
%% @todo state = undefined | {wait_for_authentication, Key} | authenticated | online
|
2011-02-13 00:53:58 +08:00
|
|
|
|
select(all) ->
|
2010-08-17 00:31:01 +08:00
|
|
|
|
List = do(qlc:q([X || X <- mnesia:table(?TABLE),
|
2010-10-11 09:20:20 +08:00
|
|
|
|
X#?TABLE.pid /= undefined
|
2010-08-17 00:31:01 +08:00
|
|
|
|
])),
|
2011-02-13 00:53:58 +08:00
|
|
|
|
{ok, List};
|
|
|
|
|
select({neighbors, User}) ->
|
2010-07-19 08:11:31 +08:00
|
|
|
|
List = do(qlc:q([X || X <- mnesia:table(?TABLE),
|
2011-02-13 07:35:05 +08:00
|
|
|
|
X#?TABLE.gid /= User#?TABLE.gid,
|
2010-08-17 00:31:01 +08:00
|
|
|
|
X#?TABLE.pid /= undefined,
|
2010-07-24 03:46:03 +08:00
|
|
|
|
X#?TABLE.instancepid =:= User#?TABLE.instancepid,
|
2010-07-19 08:11:31 +08:00
|
|
|
|
X#?TABLE.area =:= User#?TABLE.area
|
|
|
|
|
])),
|
2011-02-19 23:05:01 +08:00
|
|
|
|
{ok, List};
|
|
|
|
|
select(UsersGID) ->
|
|
|
|
|
L = [read(GID) || GID <- UsersGID],
|
|
|
|
|
[User || {ok, User} <- L].
|
2010-07-19 08:11:31 +08:00
|
|
|
|
|
2011-02-13 00:53:58 +08:00
|
|
|
|
%% @spec write(User) -> ok
|
|
|
|
|
write(User) ->
|
|
|
|
|
mnesia:transaction(fun() -> mnesia:write(User) end).
|
|
|
|
|
|
|
|
|
|
%% @spec delete(ID) -> ok
|
|
|
|
|
delete(ID) ->
|
|
|
|
|
mnesia:transaction(fun() -> mnesia:delete({?TABLE, ID}) end).
|
|
|
|
|
|
|
|
|
|
item_nth(GID, ItemIndex) ->
|
2010-09-29 05:53:41 +08:00
|
|
|
|
{atomic, [User]} = mnesia:transaction(fun() -> mnesia:read({?TABLE, GID}) end),
|
2011-02-13 00:53:58 +08:00
|
|
|
|
lists:nth(ItemIndex + 1, (User#users.character)#characters.inventory).
|
2010-09-29 05:53:41 +08:00
|
|
|
|
|
2011-02-13 00:53:58 +08:00
|
|
|
|
item_add(GID, ItemID, Variables) ->
|
2010-09-30 06:32:32 +08:00
|
|
|
|
{atomic, [User]} = mnesia:transaction(fun() -> mnesia:read({?TABLE, GID}) end),
|
2011-02-13 00:53:58 +08:00
|
|
|
|
Character = User#users.character,
|
2010-09-30 06:32:32 +08:00
|
|
|
|
Inventory = Character#characters.inventory,
|
|
|
|
|
Inventory2 = case Variables of
|
|
|
|
|
#psu_consumable_item_variables{quantity=Quantity} ->
|
2010-10-08 23:30:24 +08:00
|
|
|
|
#psu_item{data=#psu_consumable_item{max_quantity=MaxQuantity}} = egs_items_db:read(ItemID),
|
2010-09-30 06:32:32 +08:00
|
|
|
|
{ItemID, #psu_consumable_item_variables{quantity=Quantity2}} = case lists:keyfind(ItemID, 1, Inventory) of
|
|
|
|
|
false -> New = true, {ItemID, #psu_consumable_item_variables{quantity=0}};
|
|
|
|
|
Tuple -> New = false, Tuple
|
|
|
|
|
end,
|
|
|
|
|
Quantity3 = Quantity + Quantity2,
|
|
|
|
|
if Quantity3 =< MaxQuantity ->
|
|
|
|
|
lists:keystore(ItemID, 1, Inventory, {ItemID, #psu_consumable_item_variables{quantity=Quantity3}})
|
|
|
|
|
end;
|
|
|
|
|
#psu_trap_item_variables{quantity=Quantity} ->
|
2010-10-08 23:30:24 +08:00
|
|
|
|
#psu_item{data=#psu_trap_item{max_quantity=MaxQuantity}} = egs_items_db:read(ItemID),
|
2010-09-30 06:32:32 +08:00
|
|
|
|
{ItemID, #psu_trap_item_variables{quantity=Quantity2}} = case lists:keyfind(ItemID, 1, Inventory) of
|
|
|
|
|
false -> New = true, {ItemID, #psu_trap_item_variables{quantity=0}};
|
|
|
|
|
Tuple -> New = false, Tuple
|
|
|
|
|
end,
|
|
|
|
|
Quantity3 = Quantity + Quantity2,
|
|
|
|
|
if Quantity3 =< MaxQuantity ->
|
|
|
|
|
lists:keystore(ItemID, 1, Inventory, {ItemID, #psu_trap_item_variables{quantity=Quantity3}})
|
|
|
|
|
end;
|
|
|
|
|
_ ->
|
|
|
|
|
New = true,
|
|
|
|
|
if length(Inventory) < 60 ->
|
|
|
|
|
Inventory ++ [{ItemID, Variables}]
|
|
|
|
|
end
|
|
|
|
|
end,
|
|
|
|
|
Character2 = Character#characters{inventory=Inventory2},
|
2011-02-13 00:53:58 +08:00
|
|
|
|
mnesia:transaction(fun() -> mnesia:write(User#users{character=Character2}) end),
|
|
|
|
|
if New =:= false -> 16#ffffffff;
|
|
|
|
|
true -> length(Inventory2)
|
|
|
|
|
end.
|
2010-07-19 08:11:31 +08:00
|
|
|
|
|
2010-09-29 05:47:10 +08:00
|
|
|
|
%% @todo Consumable items.
|
2011-02-13 00:53:58 +08:00
|
|
|
|
item_qty_add(GID, ItemIndex, QuantityDiff) ->
|
2010-09-29 05:47:10 +08:00
|
|
|
|
{atomic, [User]} = mnesia:transaction(fun() -> mnesia:read({?TABLE, GID}) end),
|
2011-02-13 00:53:58 +08:00
|
|
|
|
Character = User#users.character,
|
2010-09-29 05:47:10 +08:00
|
|
|
|
Inventory = Character#characters.inventory,
|
|
|
|
|
{ItemID, Variables} = lists:nth(ItemIndex + 1, Inventory),
|
|
|
|
|
case Variables of
|
|
|
|
|
#psu_trap_item_variables{quantity=Quantity} ->
|
2010-10-08 23:30:24 +08:00
|
|
|
|
#psu_item{data=#psu_trap_item{max_quantity=MaxQuantity}} = egs_items_db:read(ItemID),
|
2010-09-29 05:47:10 +08:00
|
|
|
|
Quantity2 = Quantity + QuantityDiff,
|
|
|
|
|
if Quantity2 =:= 0 ->
|
|
|
|
|
Inventory2 = string:substr(Inventory, 1, ItemIndex) ++ string:substr(Inventory, ItemIndex + 2);
|
|
|
|
|
Quantity2 > 0, Quantity2 =< MaxQuantity ->
|
|
|
|
|
Variables2 = Variables#psu_trap_item_variables{quantity=Quantity2},
|
|
|
|
|
Inventory2 = string:substr(Inventory, 1, ItemIndex) ++ [{ItemID, Variables2}] ++ string:substr(Inventory, ItemIndex + 2)
|
|
|
|
|
end
|
|
|
|
|
end,
|
|
|
|
|
Character2 = Character#characters{inventory=Inventory2},
|
2011-02-13 00:53:58 +08:00
|
|
|
|
mnesia:transaction(fun() -> mnesia:write(User#users{character=Character2}) end).
|
2010-09-29 05:47:10 +08:00
|
|
|
|
|
2011-02-13 00:53:58 +08:00
|
|
|
|
shop_enter(GID, ShopID) ->
|
2010-09-30 01:02:20 +08:00
|
|
|
|
mnesia:transaction(fun() ->
|
|
|
|
|
[User] = mnesia:wread({?TABLE, GID}),
|
2011-02-13 00:53:58 +08:00
|
|
|
|
mnesia:write(User#users{shopid=ShopID})
|
|
|
|
|
end).
|
2010-09-30 01:02:20 +08:00
|
|
|
|
|
2011-02-13 00:53:58 +08:00
|
|
|
|
shop_leave(GID) ->
|
2010-09-30 01:02:20 +08:00
|
|
|
|
mnesia:transaction(fun() ->
|
|
|
|
|
[User] = mnesia:wread({?TABLE, GID}),
|
2011-02-13 00:53:58 +08:00
|
|
|
|
mnesia:write(User#users{shopid=undefined})
|
|
|
|
|
end).
|
2010-09-30 01:02:20 +08:00
|
|
|
|
|
2011-02-13 00:53:58 +08:00
|
|
|
|
shop_get(GID) ->
|
|
|
|
|
{atomic, [User]} = mnesia:transaction(fun() -> mnesia:read({?TABLE, GID}) end),
|
|
|
|
|
User#users.shopid.
|
|
|
|
|
|
|
|
|
|
money_add(GID, MoneyDiff) ->
|
2010-09-30 06:32:32 +08:00
|
|
|
|
mnesia:transaction(fun() ->
|
|
|
|
|
[User] = mnesia:wread({?TABLE, GID}),
|
2011-02-13 00:53:58 +08:00
|
|
|
|
Character = User#users.character,
|
2010-09-30 06:32:32 +08:00
|
|
|
|
Money = Character#characters.money + MoneyDiff,
|
|
|
|
|
if Money >= 0 ->
|
|
|
|
|
Character2 = Character#characters{money=Money},
|
2011-02-13 00:53:58 +08:00
|
|
|
|
mnesia:write(User#users{character=Character2})
|
2010-09-30 06:32:32 +08:00
|
|
|
|
end
|
2011-02-13 00:53:58 +08:00
|
|
|
|
end).
|
2011-02-19 22:26:54 +08:00
|
|
|
|
|
|
|
|
|
broadcast_spawn(GID, PlayersGID) ->
|
|
|
|
|
{ok, OrigUser} = read(GID),
|
|
|
|
|
lists:foreach(fun(DestGID) ->
|
|
|
|
|
{ok, DestUser} = read(DestGID),
|
|
|
|
|
DestUser#users.pid ! {egs, player_spawn, OrigUser}
|
|
|
|
|
end, PlayersGID).
|
|
|
|
|
|
|
|
|
|
broadcast_unspawn(GID, PlayersGID) ->
|
|
|
|
|
{ok, OrigUser} = read(GID),
|
|
|
|
|
lists:foreach(fun(DestGID) ->
|
|
|
|
|
{ok, DestUser} = read(DestGID),
|
|
|
|
|
DestUser#users.pid ! {egs, player_unspawn, OrigUser}
|
|
|
|
|
end, PlayersGID).
|
2011-02-20 02:45:50 +08:00
|
|
|
|
|
|
|
|
|
set_zone(GID, ZonePid, LID) ->
|
|
|
|
|
{ok, User} = read(GID),
|
|
|
|
|
write(User#users{zonepid=ZonePid, lid=LID}).
|