/*==LICENSE==*

CyanWorlds.com Engine - MMOG client, server and tools
Copyright (C) 2011 Cyan Worlds, Inc.

This program 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.

This program 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 this program. If not, see <http://www.gnu.org/licenses/>.

Additional permissions under GNU GPL version 3 section 7

If you modify this Program, or any covered work, by linking or
combining it with any of RAD Game Tools Bink SDK, Autodesk 3ds Max SDK,
NVIDIA PhysX SDK, Microsoft DirectX SDK, OpenSSL library, Independent
JPEG Group JPEG library, Microsoft Windows Media SDK, or Apple QuickTime SDK
(or a modified version of those libraries),
containing parts covered by the terms of the Bink SDK EULA, 3ds Max EULA,
PhysX SDK EULA, DirectX SDK EULA, OpenSSL and SSLeay licenses, IJG
JPEG Library README, Windows Media SDK EULA, or QuickTime SDK EULA, the
licensors of this Program grant you additional
permission to convey the resulting work. Corresponding Source for a
non-source form of such a combination shall include the source code for
the parts of OpenSSL and IJG JPEG Library used as well as that of the covered
work.

You can contact Cyan Worlds, Inc. by email legal@cyan.com
 or by snail mail at:
      Cyan Worlds, Inc.
      14617 N Newport Hwy
      Mead, WA   99021

*==LICENSE==*/
/** \file plAvatarMgr.h
	Gathering place for global animations and miscellaneous avatar data.
	Ideally this stuff will all be migrated into the resource manager. */
#ifndef PLAVATARMGR_INC
#define PLAVATARMGR_INC

#include "hsStlUtils.h"
#include "hsStlSortUtils.h"
#include "hsGeometry3.h"

#include "../pnKeyedObject/hsKeyedObject.h"
#include "../plMessage/plLoadAvatarMsg.h"

// This is still pretty much a hack, but it's a compartmentalized hack instead of the previous
// interwoven spaghetti hack.

class plSeekPointMod;
class plOneShotMod;
class plAGMasterMod;
class plArmatureMod;
class plSpawnModifier;
class plKey;
class plLoadAvatarMsg;
class plMaintainersMarkerModifier;
class plDniCoordinateInfo;
class plAvCoopMsg;
class plNotifyMsg;
class plCoopCoordinator;
class plLoadCloneMsg;
class plStatusLog;

/** \class plAvatarMgr
	Gathering place for global animations and miscellaneous avatar data.
	Ideally this stuff will all be migrated into the resource manager.
	This class is 100% static and must be explictly cleared when
	resetting the scene on shutdown or when rebuilding the scene for export.
*/
class plAvatarMgr : public hsKeyedObject
{
public:
	typedef std::vector<plArmatureMod*> plArmatureModPtrVec;

	enum AvatarTypeMask
	{
		Human = 1,
		Player = 2
	};

	plAvatarMgr();						// can only be constructed by itself (singleton)
	virtual ~plAvatarMgr();

	CLASSNAME_REGISTER( plAvatarMgr );
	GETINTERFACE_ANY( plAvatarMgr, hsKeyedObject );

	// \{
	/** Seek points are alignment points used for aligning
		the avatar before playing a detail interaction animation.
		These are registered by name here primarily for debugging
		and ad-hoc scripting. In final releases, we'll be able to
		do away with this bookeeping entirely. */
	void AddSeekPoint(plSeekPointMod *seekpoint);
	void RemoveSeekPoint(plSeekPointMod *seekpoint);
	plSeekPointMod *FindSeekPoint(const char *name);
	// \}

	// \{
	/** One shots are registered here for debugging and ad-hoc
		scripting only. */
	void AddOneShot(plOneShotMod *oneshot);
	void RemoveOneShot(plOneShotMod *oneshot);
	plOneShotMod *FindOneShot(char *name);
	// \}
	
	plKey LoadPlayer(const char* name, const char *account);
	plKey LoadPlayer(const char* name, const char *account, const char *linkName);
	plKey LoadAvatar(const char *name, const char *accountName, bool isPlayer, plKey spawnPoint, plAvTask *initialTask, const char *userStr = nil);
	/** Unload an avatar - player or npc - both locally and remotely. */
	void UnLoadAvatar(plKey avKey, bool isPlayer);
	/** send our (already loaded) local player to newly-associated clients - used when linking */
	void PropagateLocalPlayer(int spawnPoint = -1);
	/** Unload our local player on other machines because we're leaving this age.
		The player will stay around on our local machine, though. */
	bool UnPropagateLocalPlayer();

	void UnLoadRemotePlayer(plKey playerKey);
	void UnLoadLocalPlayer();

	void AddAvatar(plArmatureMod *avatar);
	void RemoveAvatar(plArmatureMod *instance);

	plArmatureMod *GetLocalAvatar();
	plKey GetLocalAvatarKey();
	static plArmatureMod *FindAvatar(plKey& avatarKey); // Key of the sceneObject
	plArmatureMod *FindAvatarByPlayerID(UInt32 pid);
	plArmatureMod *FindAvatarByModelName(char *name); // Probably only useful for custom NPCs. All players are
													  // either "Male" or "Female".
	void FindAllAvatarsByModelName(const char* name, plArmatureModPtrVec& outVec);
	plArmatureMod *GetFirstRemoteAvatar();

	// \{
	/** Spawn points are potential entry points for the 
		avatar. They're selected pretty randomly right now;
		eventually they'll be selected by script based
		on the book used to enter the scene. */
	void AddSpawnPoint(plSpawnModifier *spawn);
	void RemoveSpawnPoint(plSpawnModifier *spawn);
	const plSpawnModifier *GetSpawnPoint(int index);
	int	NumSpawnPoints() { return fSpawnPoints.size(); }
	int	FindSpawnPoint( const char *name ) const;
	// \}
	static int WarpPlayerToAnother(hsBool iMove, UInt32 remoteID);
	static int WarpPlayerToXYZ(hsScalar x, hsScalar y, hsScalar z);
	static int WarpPlayerToXYZ(int pid, hsScalar x, hsScalar y, hsScalar z);

	static plAvatarMgr *GetInstance();
	static void ShutDown();


	hsBool MsgReceive(plMessage *msg);
	hsBool HandleCoopMsg(plAvCoopMsg *msg);
	hsBool HandleNotifyMsg(plNotifyMsg *msg);
	hsBool IPassMessageToActiveCoop(plMessage *msg, UInt32 id, UInt16 serial);

	// similar to a spawn point, maintainers markers are used 
	// to generate your position in Dni coordinates
	void AddMaintainersMarker(plMaintainersMarkerModifier *mm);
	void RemoveMaintainersMarker(plMaintainersMarkerModifier *mm);
	void PointToDniCoordinate(hsPoint3 pt, plDniCoordinateInfo* ret);
	void GetDniCoordinate(plDniCoordinateInfo* ret);

	static void OfferLinkingBook(plKey hostKey, plKey guestKey, plMessage *linkMsg, plKey replyKey);

	bool IsACoopRunning();
	plStatusLog *GetLog() { return fLog; }

protected:
	/** Dump all internal data. */
	void IReset();
	
	/** Handle an incoming clone message; do any necessary post-processing
		on the avatar. */
	void plAvatarMgr::IFinishLoadingAvatar(plLoadAvatarMsg *cloneMsg);

	/** Handle an incoming clone message which holds an unload request.
	*/
	void plAvatarMgr::IFinishUnloadingAvatar(plLoadAvatarMsg *cloneMsg);
	
	/** When an armature modifier attached to the given scene object is loaded,
		send it the given message.
		We get notified when the avatar's scene object is loaded, but we also need to 
		set some information up for the avatar modifier when it comes in.
		We'll get that notification via the AddAvatar call later. In this function
		we're going to squirrel away an initialization message to pass to the armature
		modifier when it arrives. */
	void plAvatarMgr::IDeferInit(plKey playerSOKey, plMessage *initMsg);
	
	/** See if we have an avatar type message saved for the given avatar and send them. */
	void plAvatarMgr::ISendDeferredInit(plKey playerSOKey);

	static plAvatarMgr*	fInstance;		// the single instance of the avatar manager

	typedef std::map<const char *, plSeekPointMod *, stringISorter> plSeekPointMap;
	plSeekPointMap fSeekPoints;

	typedef std::map<char *, plOneShotMod *, stringISorter> plOneShotMap;
	plOneShotMap fOneShots;

	typedef std::map<plKey, plMessage *> DeferredInits;
	DeferredInits fDeferredInits;

//	typedef std::map<const char *, plArmatureMod *, stringISorter> plAvatarMap;
	typedef std::vector<plKey> plAvatarVec;
	plAvatarVec fAvatars;

	typedef std::vector<const plSpawnModifier*> plSpawnVec;
	plSpawnVec	fSpawnPoints;

	hsTArray<plMaintainersMarkerModifier*> fMaintainersMarkers;

	// we're using a multimap, which is a map which allows multiple entries to
	// share the same key. the key we use is the initiator's player id; in the vast
	// majority of cases, there will only be one coop running for a given initiator's
	// ID. By using a multimap, however, we can still handle a few different coops
	// for the same user by just iterating from the first match forward until
	// we run out of matches.
	typedef std::multimap<UInt32, plCoopCoordinator *> plCoopMap;
	plCoopMap fActiveCoops;

	hsTArray<plLoadCloneMsg*> fCloneMsgQueue;
	plStatusLog *fLog;	
};


#endif // PLAVATARMGR_INC