dsI/bin/
dsI/extra/creremote/
dsI/extra/mingw/
dsI/extra/wolfpaw/
dsI/fluffos-2.7-ds2.018/
dsI/fluffos-2.7-ds2.018/ChangeLog.old/
dsI/fluffos-2.7-ds2.018/Win32/
dsI/fluffos-2.7-ds2.018/compat/
dsI/fluffos-2.7-ds2.018/compat/simuls/
dsI/fluffos-2.7-ds2.018/testsuite/
dsI/fluffos-2.7-ds2.018/testsuite/clone/
dsI/fluffos-2.7-ds2.018/testsuite/command/
dsI/fluffos-2.7-ds2.018/testsuite/data/
dsI/fluffos-2.7-ds2.018/testsuite/etc/
dsI/fluffos-2.7-ds2.018/testsuite/include/
dsI/fluffos-2.7-ds2.018/testsuite/inherit/
dsI/fluffos-2.7-ds2.018/testsuite/inherit/master/
dsI/fluffos-2.7-ds2.018/testsuite/log/
dsI/fluffos-2.7-ds2.018/testsuite/single/
dsI/fluffos-2.7-ds2.018/testsuite/single/tests/compiler/
dsI/fluffos-2.7-ds2.018/testsuite/single/tests/efuns/
dsI/fluffos-2.7-ds2.018/testsuite/single/tests/operators/
dsI/fluffos-2.7-ds2.018/testsuite/u/
dsI/fluffos-2.7-ds2.018/tmp/
dsI/lib/cfg/
dsI/lib/cmds/common/
dsI/lib/cmds/creators/include/
dsI/lib/cmds/creators/include/SCCS/
dsI/lib/daemon/services/
dsI/lib/doc/
dsI/lib/domains/Ylsrim/
dsI/lib/domains/Ylsrim/adm/
dsI/lib/domains/Ylsrim/armour/
dsI/lib/domains/Ylsrim/broken/
dsI/lib/domains/Ylsrim/fish/
dsI/lib/domains/Ylsrim/meal/
dsI/lib/domains/Ylsrim/npc/
dsI/lib/domains/Ylsrim/virtual/
dsI/lib/domains/Ylsrim/weapon/
dsI/lib/domains/default/creator/
dsI/lib/domains/default/etc/
dsI/lib/domains/default/room/
dsI/lib/lib/comp/
dsI/lib/lib/lvs/
dsI/lib/lib/user/
dsI/lib/lib/virtual/
dsI/lib/obj/
dsI/lib/obj/include/
dsI/lib/realms/
dsI/lib/save/kills/a/
dsI/lib/save/kills/b/
dsI/lib/save/kills/f/
dsI/lib/save/kills/m/
dsI/lib/save/kills/q/
dsI/lib/save/kills/r/
dsI/lib/secure/cfg/
dsI/lib/secure/cfg/classes/
dsI/lib/secure/cfg/races/SCCS/
dsI/lib/secure/cmds/creators/include/
dsI/lib/secure/cmds/players/
dsI/lib/secure/cmds/players/include/
dsI/lib/secure/daemon/include/
dsI/lib/secure/lib/
dsI/lib/secure/lib/include/
dsI/lib/secure/lib/net/
dsI/lib/secure/lib/net/include/
dsI/lib/secure/lib/std/
dsI/lib/secure/obj/
dsI/lib/secure/obj/include/
dsI/lib/secure/save/
dsI/lib/spells/
dsI/lib/verbs/admins/include/
dsI/lib/verbs/common/
dsI/lib/verbs/common/include/
dsI/lib/verbs/creators/
dsI/lib/verbs/creators/include/
dsI/lib/verbs/players/include/SCCS/
dsI/lib/verbs/rooms/
dsI/lib/verbs/rooms/include/
dsI/lib/www/
dsI/v22.2b14/
dsI/win32/
/*    /daemon/party.c
 *    from the Dead Souls  Object Library
 *    manages parties
 *    created by Descartes of Borg 951120
 */

#include <lib.h>
#include <daemons.h>
#include <message_class.h>
#include "include/party.h"

inherit LIB_DAEMON;

mapping Parties;

static void create() {
    daemon::create();
    SetNoClean(1);
    Parties = ([]);
}

mixed CanChangeLeader(object who, object targ) {
    class party p;
    string pname;

    if( !(p = Parties[pname]) ) return "No such party exists.";
    if( p->Leader != who )
      return "You must be the party leader in order to change leaders.";
    if( member_array(targ, p->Members) == -1 )
      return (string)targ->GetName() + " is not in the party.";
    return 1;
}

mixed CanCreateParty(object who, string name) {
    if( Parties[name] ) return "A party by that name already exists.";
    if( member_array(name, (string *)CHAT_D->GetChannels()) != -1 )
      return "You cannot use the name " + name + " for your party.";
    if( (string)who->GetParty() ) return "You are already in a party!";
    return 1;
}

mixed CanInviteMember(object who, object member) {
    string pname;

    pname = (string)who->GetParty();
    if( !Parties[pname] ) return "There is no such party!";
    if( ((class party)Parties[pname])->Leader != who )
      return "Only the party leader may add members!";
    if( (string)member->GetParty() )
      return (string)member->GetName() + " is already in a party.";
    if( environment(member) != environment(who) )
      return (string)member->GetName() + " must be somewhere near you.";
    return 1;
}

mixed CanJoinParty(object who, string pname) {
    if( !Parties[pname] ) return "There is no such party.";
    if( member_array(who, ((class party)Parties[pname])->Invited) == -1 )
      return "You have not been invited to join that party.";
    if( (string)who->GetParty() ) return "You are already in a party.";
    if( environment(who) != environment(((class party)Parties[pname])->Leader) )
      return "You are nowhere the leader of the party.";
    return 1;
}

mixed CanLeaveParty(object who) {
    class party p;
    string pname;

    pname = (string)who->GetParty();
    if( !pname || !(p = Parties[pname]) ) return "There is no such party.";
    if( member_array(who, p->Members) == -1 )
      return "You are not in that party.";
    return 1;
}

mixed CanRemoveMember(object who, object targ) {
    class party p;
    string pname;

    pname = (string)who->GetParty();
    if( !(p = Parties[pname]) ) return "There is no such party.";
    if( p->Leader != who ) return "Only the party leader may remove people.";
     return 1;
}	

mixed CanRemoveParty(object who) {
    class party p;
    string pname;

    pname = (string)who->GetParty();
    if( !(p = Parties[pname]) ) return "There is no such party!";
    if( p->Leader != who )
      return "Only the party leader may disband the party.";
    return 1;
}

mixed eventChangeLeader(object who, object targ) {
    class party p;
    string pname;

    pname = (string)who->GetParty();
    p = Parties[pname];
    p->Leader = targ;
    CHAT_D->eventSend("System", pname, (string)targ->GetName() + " is now "
		      "the leader.");
    return 1;
}

mixed eventCreateParty(object who, string name) {
    class party this_party;

    if( (string)who->SetParty(name) != name )
      return "There was some bizarre problem sticking you in a party.";
    this_party = new(class party);
    this_party->Leader = who;
    this_party->Members = ({ who });
    this_party->Invited = ({});
    Parties[name] = this_party;
    who->eventPrint("Party " + name + " successfully created.", MSG_SYSTEM);
    return 1;
}

mixed eventInviteMember(object who, object targ) {
    class party this_party;
    mixed tmp;
    string name;

    name = (string)who->GetParty();
    this_party = Parties[name];
    this_party->Invited += ({ targ });
    CHAT_D->eventSendChannel("System", name, (string)targ->GetName() +
			     " has been invited to join the party.");    
    call_out((: RemoveInvitiation :), 60, name, targ);
    targ->eventPrint("You have been invited to join the party \"" + name +
		     "\".\nType \"party join " + name + "\" in 60 "
		     "seconds to join.", MSG_SYSTEM);
    return 1;
}

mixed eventJoinParty(object who, string name) {
    class party this_party;
    mixed tmp;
    
    if( (tmp = CanJoinParty(who, name)) != 1 ) return tmp;
    this_party = Parties[name];
    if( (string)who->SetParty(name) != name )
      return "Bogus error in joining party.";
    this_party->Invited -= ({ who });
    this_party->Members += ({ who });
    CHAT_D->eventSendChannel("System", name, (string)who->GetName() +
			     " has joined the party.");    
    return 1;
}

mixed eventLeaveParty(object who) {
    return eventRemoveMember(who, who);
}

mixed eventRemoveMember(object who, object targ) {
    class party p;
    object ob = 0;
    string name;

    p = Parties[name = (string)who->GetParty()];
    if( targ == p->Leader ) {
	if( sizeof(p->Members) == 1 ) return eventRemoveParty(who);
	else {
	    foreach(ob in p->Members) if( ob != targ ) break;
	    if( !ob ) return eventRemoveParty(who);
	    else {
		p->Leader = ob;
		ob->eventPrint("You are now the leader of the party " + name +
			       ".", MSG_SYSTEM);
	    }
	}
    }
    targ->SetParty(0);
    if( Parties[name] ) {
	p->Members -= ({ targ });
	CHAT_D->eventSendChannel("System", name, (string)targ->GetName() +
				 " is no longer in the party.");
    }
    targ->eventPrint("You are no longer a member of the party " + name +
		     ".", MSG_SYSTEM);
    return 1;
}

mixed eventRemoveParty(object who) {
    object ob;
    string name;

    name = (string)who->GetParty();
    CHAT_D->eventSendChannel("System", name, "The party " + name + " has been "
			     "disbanded.");
    foreach(ob in ((class party)Parties[name])->Members)
      ob->SetParty(0);
    map_delete(Parties, name);
    return 1;
}

object GetPartyLeader(string name) {
    class party p;
    string nom;
    
    foreach(nom, p in Parties) if( nom == name ) return p->Leader;
    return 0;
}

object *GetPartyMembers(string name) {
    class party p;
    string nom;

    foreach(nom, p in Parties) if( nom == name ) return p->Members;
    return 0;
}

static void RemoveInvitiation(string name, object who) {
    if( !Parties[name] ) return;
    if( member_array(who, ((class party)Parties[name])->Invited) == -1 )
      return;
    ((class party)Parties[name])->Invited -= ({ who });
}