|
|
|
/*==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==*/
|
|
|
|
#include "plNetLinkingMgr.h"
|
|
|
|
#include "plNetClientMgr.h"
|
|
|
|
#include "plNetCliAgeJoiner.h"
|
|
|
|
#include "plNetCliAgeLeaver.h"
|
|
|
|
|
|
|
|
#include "plNetTransport/plNetTransportMember.h" // OfferLinkToPlayer()
|
|
|
|
|
|
|
|
#include "plgDispatch.h"
|
|
|
|
#include "pnMessage/plTimeMsg.h"
|
|
|
|
#include "plMessage/plLinkToAgeMsg.h"
|
|
|
|
#include "pnKeyedObject/plKey.h"
|
|
|
|
#include "pnKeyedObject/plUoid.h"
|
|
|
|
#include "pnKeyedObject/hsKeyedObject.h"
|
|
|
|
#include "pnSceneObject/plSceneObject.h"
|
|
|
|
#include "plNetCommon/plNetCommon.h"
|
|
|
|
#include "plVault/plVault.h"
|
|
|
|
#include "pnNetCommon/pnNetCommon.h"
|
|
|
|
#include "plMessage/plVaultNotifyMsg.h"
|
|
|
|
#include "plNetMessage/plNetMessage.h"
|
|
|
|
#include "plAvatar/plAvatarMgr.h"
|
|
|
|
#include "plAvatar/plArmatureMod.h"
|
|
|
|
#include "plFile/hsFiles.h"
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
|
|
*
|
|
|
|
* Private
|
|
|
|
*
|
|
|
|
***/
|
|
|
|
|
|
|
|
struct NlmOpNode;
|
|
|
|
|
|
|
|
enum ENlmOp {
|
|
|
|
kNlmOpNoOp,
|
|
|
|
kNlmOpWaitOp,
|
|
|
|
kNlmOpJoinAgeOp,
|
|
|
|
kNlmOpLeaveAgeOp,
|
|
|
|
kNumNlmOps
|
|
|
|
};
|
|
|
|
|
|
|
|
struct NlmOp {
|
|
|
|
ENlmOp opcode;
|
|
|
|
NlmOpNode * node;
|
|
|
|
NlmOp (const ENlmOp & op)
|
|
|
|
: opcode(op)
|
|
|
|
{ }
|
|
|
|
};
|
|
|
|
|
|
|
|
struct NlmNoOpOp : NlmOp {
|
|
|
|
NlmNoOpOp ()
|
|
|
|
: NlmOp(kNlmOpNoOp)
|
|
|
|
{ }
|
|
|
|
};
|
|
|
|
|
|
|
|
struct NlmOpWaitOp : NlmOp {
|
|
|
|
NlmOpWaitOp ()
|
|
|
|
: NlmOp(kNlmOpWaitOp)
|
|
|
|
{ }
|
|
|
|
};
|
|
|
|
|
|
|
|
struct NlmJoinAgeOp : NlmOp {
|
|
|
|
NetCommAge age;
|
|
|
|
NlmJoinAgeOp ()
|
|
|
|
: NlmOp(kNlmOpJoinAgeOp)
|
|
|
|
{ }
|
|
|
|
};
|
|
|
|
|
|
|
|
struct NlmLeaveAgeOp : NlmOp {
|
|
|
|
bool quitting;
|
|
|
|
NlmLeaveAgeOp ()
|
|
|
|
: NlmOp(kNlmOpLeaveAgeOp)
|
|
|
|
{ }
|
|
|
|
};
|
|
|
|
|
|
|
|
struct NlmOpNode {
|
|
|
|
LINK(NlmOpNode) link;
|
|
|
|
NlmOp * op;
|
|
|
|
~NlmOpNode () { DEL(op); }
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
|
|
*
|
|
|
|
* Private data
|
|
|
|
*
|
|
|
|
***/
|
|
|
|
|
|
|
|
static plNCAgeJoiner * s_ageJoiner;
|
|
|
|
static plNCAgeLeaver * s_ageLeaver;
|
|
|
|
static LISTDECL(NlmOpNode, link) s_oplist;
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
|
|
*
|
|
|
|
* Local functions
|
|
|
|
*
|
|
|
|
***/
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
static void QueueOp (NlmOp * op, bool front = false) {
|
|
|
|
NlmOpNode * node = NEWZERO(NlmOpNode);
|
|
|
|
node->op = op;
|
|
|
|
op->node = node;
|
|
|
|
s_oplist.Link(node, front ? kListHead : kListTail);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
|
|
*
|
|
|
|
* plNetLinkingMgr
|
|
|
|
*
|
|
|
|
***/
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
void plNetLinkingMgr::NCAgeJoinerCallback (
|
|
|
|
plNCAgeJoiner * joiner,
|
|
|
|
unsigned type,
|
|
|
|
void * notify,
|
|
|
|
void * userState
|
|
|
|
) {
|
|
|
|
plNetLinkingMgr * lm = plNetLinkingMgr::GetInstance();
|
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
case kAgeJoinerComplete: {
|
|
|
|
ASSERT(joiner == s_ageJoiner);
|
|
|
|
s_ageJoiner = nil;
|
|
|
|
|
|
|
|
lm->IPostProcessLink();
|
|
|
|
|
|
|
|
lm->fLinkedIn = true;
|
|
|
|
lm->SetEnabled(true);
|
|
|
|
|
|
|
|
// Pull our wait op off exec queue
|
|
|
|
if (NlmOpWaitOp * waitOp = (NlmOpWaitOp *) userState)
|
|
|
|
DEL(waitOp->node);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
DEFAULT_FATAL(type);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
void plNetLinkingMgr::NCAgeLeaverCallback (
|
|
|
|
plNCAgeLeaver * leaver,
|
|
|
|
unsigned type,
|
|
|
|
void * notify,
|
|
|
|
void * userState
|
|
|
|
) {
|
|
|
|
switch (type) {
|
|
|
|
case kAgeLeaveComplete: {
|
|
|
|
ASSERT(leaver == s_ageLeaver);
|
|
|
|
s_ageLeaver = nil;
|
|
|
|
|
|
|
|
// Pull our wait op off exec queue
|
|
|
|
if (NlmOpWaitOp * waitOp = (NlmOpWaitOp *) userState)
|
|
|
|
DEL(waitOp->node);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
DEFAULT_FATAL(type);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
void plNetLinkingMgr::ExecNextOp () {
|
|
|
|
plNetLinkingMgr * lm = plNetLinkingMgr::GetInstance();
|
|
|
|
|
|
|
|
NlmOpNode * opNode = s_oplist.Head();
|
|
|
|
if (!opNode)
|
|
|
|
return;
|
|
|
|
|
|
|
|
switch (opNode->op->opcode) {
|
|
|
|
case kNlmOpNoOp: {
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case kNlmOpWaitOp: {
|
|
|
|
}
|
|
|
|
return; // don't allow wait op to be unlinked/deleted from list
|
|
|
|
|
|
|
|
case kNlmOpJoinAgeOp: {
|
|
|
|
ASSERT(!s_ageJoiner);
|
|
|
|
ASSERT(!s_ageLeaver);
|
|
|
|
|
|
|
|
// Insert a wait operation into the exec queue
|
|
|
|
NlmOpWaitOp * waitOp = NEWZERO(NlmOpWaitOp);
|
|
|
|
QueueOp(waitOp, true);
|
|
|
|
|
|
|
|
NlmJoinAgeOp * joinAgeOp = (NlmJoinAgeOp *) opNode->op;
|
|
|
|
NCAgeJoinerCreate(
|
|
|
|
&s_ageJoiner,
|
|
|
|
joinAgeOp->age,
|
|
|
|
NCAgeJoinerCallback,
|
|
|
|
waitOp
|
|
|
|
);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case kNlmOpLeaveAgeOp: {
|
|
|
|
ASSERT(!s_ageJoiner);
|
|
|
|
ASSERT(!s_ageLeaver);
|
|
|
|
|
|
|
|
// Insert a wait operation into the exec queue
|
|
|
|
NlmOpWaitOp * waitOp = NEWZERO(NlmOpWaitOp);
|
|
|
|
QueueOp(waitOp, true);
|
|
|
|
|
|
|
|
lm->SetEnabled(false);
|
|
|
|
lm->fLinkedIn = false;
|
|
|
|
|
|
|
|
NlmLeaveAgeOp * leaveAgeOp = (NlmLeaveAgeOp *) opNode->op;
|
|
|
|
NCAgeLeaverCreate(
|
|
|
|
&s_ageLeaver,
|
|
|
|
leaveAgeOp->quitting,
|
|
|
|
NCAgeLeaverCallback,
|
|
|
|
waitOp
|
|
|
|
);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
DEL(opNode);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
plNetLinkingMgr::plNetLinkingMgr()
|
|
|
|
: fLinkingEnabled(true)
|
|
|
|
, fLinkedIn (false)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
plNetLinkingMgr * plNetLinkingMgr::GetInstance()
|
|
|
|
{
|
|
|
|
static plNetLinkingMgr Instance;
|
|
|
|
return &Instance;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void plNetLinkingMgr::SetEnabled( bool b )
|
|
|
|
{
|
|
|
|
plNetClientMgr * nc = plNetClientMgr::GetInstance();
|
|
|
|
hsLogEntry( nc->DebugMsg( "plNetLinkingMgr: %s -> %s", fLinkingEnabled?"Enabled":"Disabled",b?"Enabled":"Disabled" ) );
|
|
|
|
fLinkingEnabled = b;
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
// static
|
|
|
|
std::string plNetLinkingMgr::GetProperAgeName( const char * ageName )
|
|
|
|
{
|
|
|
|
plNetClientMgr * nc = plNetClientMgr::GetInstance();
|
|
|
|
hsFolderIterator it("dat"PATH_SEPARATOR_STR"*.age", true);
|
|
|
|
while ( it.NextFile() )
|
|
|
|
{
|
|
|
|
std::string work = it.GetFileName();
|
|
|
|
work.erase( work.find( ".age" ) );
|
|
|
|
if ( stricmp( ageName, work.c_str() )==0 )
|
|
|
|
return work;
|
|
|
|
}
|
|
|
|
return ageName;
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
hsBool plNetLinkingMgr::MsgReceive( plMessage *msg )
|
|
|
|
{
|
|
|
|
if (s_ageLeaver && NCAgeLeaverMsgReceive(s_ageLeaver, msg))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (s_ageJoiner && NCAgeJoinerMsgReceive(s_ageJoiner, msg))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (plLinkToAgeMsg * pLinkMsg = plLinkToAgeMsg::ConvertNoRef(msg)) {
|
|
|
|
if (!fLinkingEnabled)
|
|
|
|
hsLogEntry(plNetClientMgr::GetInstance()->DebugMsg("Not linking. Linking is disabled."));
|
|
|
|
else
|
|
|
|
IProcessLinkToAgeMsg(pLinkMsg);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (plLinkingMgrMsg * pLinkingMgrMsg = plLinkingMgrMsg::ConvertNoRef(msg)) {
|
|
|
|
IProcessLinkingMgrMsg( pLinkingMgrMsg );
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If a link was deferred in order to register an owned age, we will
|
|
|
|
// get a VaultNotify about the registration
|
|
|
|
if (plVaultNotifyMsg* vaultMsg = plVaultNotifyMsg::ConvertNoRef(msg)) {
|
|
|
|
IProcessVaultNotifyMsg(vaultMsg);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void plNetLinkingMgr::Update()
|
|
|
|
{
|
|
|
|
if (s_ageLeaver)
|
|
|
|
NCAgeLeaverUpdate(s_ageLeaver);
|
|
|
|
if (s_ageJoiner)
|
|
|
|
NCAgeJoinerUpdate(s_ageJoiner);
|
|
|
|
|
|
|
|
ExecNextOp();
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
bool plNetLinkingMgr::IProcessLinkToAgeMsg( plLinkToAgeMsg * msg )
|
|
|
|
{
|
|
|
|
if (!fLinkingEnabled) {
|
|
|
|
hsLogEntry( plNetClientMgr::GetInstance()->DebugMsg( "Not linking. Linking is disabled." ) );
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
plNetClientMgr * nc = plNetClientMgr::GetInstance();
|
|
|
|
|
|
|
|
bool result = true;
|
|
|
|
|
|
|
|
plAgeLinkStruct save;
|
|
|
|
save.CopyFrom( GetPrevAgeLink() );
|
|
|
|
GetPrevAgeLink()->CopyFrom( GetAgeLink() );
|
|
|
|
GetAgeLink()->CopyFrom( msg->GetAgeLink() );
|
|
|
|
|
|
|
|
// Actually do stuff...
|
|
|
|
UInt8 pre = IPreProcessLink();
|
|
|
|
if (pre == kLinkImmediately)
|
|
|
|
{
|
|
|
|
msg->Ref();
|
|
|
|
IDoLink(msg);
|
|
|
|
}
|
|
|
|
else if (pre == kLinkDeferred)
|
|
|
|
{
|
|
|
|
msg->Ref();
|
|
|
|
fDeferredLink = msg;
|
|
|
|
}
|
|
|
|
else if (pre == kLinkFailed)
|
|
|
|
{
|
|
|
|
hsLogEntry( nc->ErrorMsg( "IPreProcessLink failed. Not linking." ) );
|
|
|
|
// Restore previous age info state.
|
|
|
|
GetAgeLink()->CopyFrom( GetPrevAgeLink() );
|
|
|
|
GetPrevAgeLink()->CopyFrom( &save );
|
|
|
|
result = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void plNetLinkingMgr::IDoLink(plLinkToAgeMsg* msg)
|
|
|
|
{
|
|
|
|
plNetClientMgr* nc = plNetClientMgr::GetInstance();
|
|
|
|
GetAgeLink()->SetSpawnPoint(msg->GetAgeLink()->SpawnPoint());
|
|
|
|
|
|
|
|
if (fLinkedIn) {
|
|
|
|
// Set the link out animation we should use
|
|
|
|
if (plSceneObject *localSO = plSceneObject::ConvertNoRef(nc->GetLocalPlayer())) {
|
|
|
|
plArmatureMod *avMod = const_cast<plArmatureMod*>(plArmatureMod::ConvertNoRef(localSO->GetModifierByType(plArmatureMod::Index())));
|
|
|
|
avMod->SetLinkInAnim(msg->GetLinkInAnimName());
|
|
|
|
}
|
|
|
|
// Queue leave op
|
|
|
|
NlmLeaveAgeOp * leaveAgeOp = NEWZERO(NlmLeaveAgeOp);
|
|
|
|
QueueOp(leaveAgeOp);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Queue join op
|
|
|
|
NlmJoinAgeOp * joinAgeOp = NEWZERO(NlmJoinAgeOp);
|
|
|
|
joinAgeOp->age.ageInstId = (Uuid) *GetAgeLink()->GetAgeInfo()->GetAgeInstanceGuid();
|
|
|
|
StrCopy(
|
|
|
|
joinAgeOp->age.ageDatasetName,
|
|
|
|
GetAgeLink()->GetAgeInfo()->GetAgeFilename(),
|
|
|
|
arrsize(joinAgeOp->age.ageDatasetName)
|
|
|
|
);
|
|
|
|
StrCopy(
|
|
|
|
joinAgeOp->age.spawnPtName,
|
|
|
|
GetAgeLink()->SpawnPoint().GetName(),
|
|
|
|
arrsize(joinAgeOp->age.spawnPtName)
|
|
|
|
);
|
|
|
|
QueueOp(joinAgeOp);
|
|
|
|
|
|
|
|
// UnRef
|
|
|
|
msg->UnRef();
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
bool plNetLinkingMgr::IProcessLinkingMgrMsg( plLinkingMgrMsg * msg )
|
|
|
|
{
|
|
|
|
plNetClientMgr * nc = plNetClientMgr::GetInstance();
|
|
|
|
|
|
|
|
bool result = true;
|
|
|
|
|
|
|
|
switch ( msg->GetCmd() )
|
|
|
|
{
|
|
|
|
/////////////////////
|
|
|
|
case kLinkPlayerHere:
|
|
|
|
{
|
|
|
|
// player wants to link to our age
|
|
|
|
UInt32 playerID = msg->GetArgs()->GetInt( 0 );
|
|
|
|
hsLogEntry( nc->DebugMsg( "Linking player %lu to this age.", playerID ) );
|
|
|
|
LinkPlayerHere( playerID );
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
/////////////////////
|
|
|
|
case kLinkPlayerToPrevAge:
|
|
|
|
{
|
|
|
|
// link myself back to my last age
|
|
|
|
hsLogEntry( nc->DebugMsg( "Linking back to my last age.") );
|
|
|
|
LinkToPrevAge();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
/////////////////////
|
|
|
|
case kOfferLinkToPlayer:
|
|
|
|
{
|
|
|
|
// // Notify the KI that we received an offer.
|
|
|
|
// plVaultNotifyMsg * notify = TRACKED_NEW plVaultNotifyMsg();
|
|
|
|
// notify->SetType( plVaultNotifyMsg::kPlayerOfferedLink );
|
|
|
|
// notify->GetArgs()->AddItem( 0, msg->GetArgs()->GetItem( 0 ), true ); // add to notify and have notify take over memory management of the item.
|
|
|
|
// msg->GetArgs()->RemoveItem( 0, true ); // msg to stop memory managing item, notify msg will delete it.
|
|
|
|
// notify->Send();
|
|
|
|
|
|
|
|
plAgeLinkStruct *myLink = plAgeLinkStruct::ConvertNoRef(msg->GetArgs()->GetItem( 0 ));
|
|
|
|
LinkToAge(myLink);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
/////////////////////
|
|
|
|
default:
|
|
|
|
hsAssert( false, "IProcessLinkingMgrMsg: Unknown linking mgr cmd." );
|
|
|
|
result = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
bool plNetLinkingMgr::IProcessVaultNotifyMsg(plVaultNotifyMsg* msg)
|
|
|
|
{
|
|
|
|
// No deferred link? Bye bye.
|
|
|
|
if (fDeferredLink == nil)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
plAgeLinkStruct* cur = GetAgeLink();
|
|
|
|
RelVaultNode* cVaultLink = nil;
|
|
|
|
switch (msg->GetType())
|
|
|
|
{
|
|
|
|
case plVaultNotifyMsg::kRegisteredChildAgeLink:
|
|
|
|
case plVaultNotifyMsg::kRegisteredOwnedAge:
|
|
|
|
case plVaultNotifyMsg::kRegisteredSubAgeLink:
|
|
|
|
cVaultLink = VaultGetNodeIncRef(msg->GetArgs()->GetInt(plNetCommon::VaultTaskArgs::kAgeLinkNode));
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cVaultLink != nil)
|
|
|
|
{
|
|
|
|
// This is something that Cyan does... >.<
|
|
|
|
// It's very useful though...
|
|
|
|
VaultAgeLinkNode accLink(cVaultLink);
|
|
|
|
accLink.CopyTo(cur);
|
|
|
|
if (RelVaultNode* rvnInfo = cVaultLink->GetChildNodeIncRef(plVault::kNodeType_AgeInfo, 1))
|
|
|
|
{
|
|
|
|
VaultAgeInfoNode accInfo(rvnInfo);
|
|
|
|
accInfo.CopyTo(cur->GetAgeInfo());
|
|
|
|
rvnInfo->DecRef();
|
|
|
|
}
|
|
|
|
|
|
|
|
IDoLink(fDeferredLink);
|
|
|
|
fDeferredLink = nil;
|
|
|
|
return true;
|
|
|
|
|
|
|
|
cVaultLink->DecRef();
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
bool plNetLinkingMgr::IDispatchMsg( plMessage * msg, UInt32 playerID )
|
|
|
|
{
|
|
|
|
plNetClientMgr * nc = plNetClientMgr::GetInstance();
|
|
|
|
|
|
|
|
msg->AddReceiver( plNetClientMgr::GetInstance()->GetKey() );
|
|
|
|
|
|
|
|
if ( playerID!=kInvalidPlayerID && playerID!=nc->GetPlayerID() )
|
|
|
|
{
|
|
|
|
msg->SetBCastFlag( plMessage::kNetAllowInterAge );
|
|
|
|
msg->SetBCastFlag( plMessage::kNetPropagate );
|
|
|
|
msg->SetBCastFlag( plMessage::kNetForce );
|
|
|
|
msg->SetBCastFlag( plMessage::kLocalPropagate, 0 );
|
|
|
|
// send msg to other player (maybe in different age than us)
|
|
|
|
msg->AddNetReceiver( playerID );
|
|
|
|
}
|
|
|
|
|
|
|
|
return ( msg->Send()!=0 );
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void plNetLinkingMgr::LinkToAge( plAgeLinkStruct * link, UInt32 playerID )
|
|
|
|
{
|
|
|
|
LinkToAge(link, nil, playerID);
|
|
|
|
}
|
|
|
|
|
|
|
|
void plNetLinkingMgr::LinkToAge( plAgeLinkStruct * link, const char* linkAnim, UInt32 playerID )
|
|
|
|
{
|
|
|
|
if ( !fLinkingEnabled )
|
|
|
|
{
|
|
|
|
hsLogEntry( plNetClientMgr::GetInstance()->DebugMsg( "Not linking. Linking is disabled." ) );
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
plLinkToAgeMsg* pMsg = TRACKED_NEW plLinkToAgeMsg( link );
|
|
|
|
if (linkAnim)
|
|
|
|
pMsg->SetLinkInAnimName(linkAnim);
|
|
|
|
IDispatchMsg( pMsg, playerID );
|
|
|
|
}
|
|
|
|
|
|
|
|
// link myself back to my last age
|
|
|
|
void plNetLinkingMgr::LinkToPrevAge( UInt32 playerID )
|
|
|
|
{
|
|
|
|
if ( !fLinkingEnabled )
|
|
|
|
{
|
|
|
|
hsLogEntry( plNetClientMgr::GetInstance()->DebugMsg( "Not linking. Linking is disabled." ) );
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (GetPrevAgeLink()->GetAgeInfo()->HasAgeFilename())
|
|
|
|
{
|
|
|
|
plLinkToAgeMsg* pMsg = TRACKED_NEW plLinkToAgeMsg( GetPrevAgeLink() );
|
|
|
|
IDispatchMsg( pMsg, playerID );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
hsLogEntry( plNetClientMgr::GetInstance()->DebugMsg( "Not linking. No prev age fileName." ) );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void plNetLinkingMgr::LinkToMyPersonalAge( UInt32 playerID )
|
|
|
|
{
|
|
|
|
if ( !fLinkingEnabled )
|
|
|
|
{
|
|
|
|
hsLogEntry( plNetClientMgr::GetInstance()->DebugMsg( "Not linking. Linking is disabled." ) );
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
plNetClientMgr* nc = plNetClientMgr::GetInstance();
|
|
|
|
|
|
|
|
plAgeLinkStruct link;
|
|
|
|
link.GetAgeInfo()->SetAgeFilename( kPersonalAgeFilename );
|
|
|
|
link.SetLinkingRules( plNetCommon::LinkingRules::kOwnedBook );
|
|
|
|
|
|
|
|
plSpawnPointInfo hutSpawnPoint;
|
|
|
|
hutSpawnPoint.SetName(kPersonalAgeLinkInPointCloset);
|
|
|
|
link.SetSpawnPoint(hutSpawnPoint);
|
|
|
|
|
|
|
|
plLinkToAgeMsg* pMsg = TRACKED_NEW plLinkToAgeMsg( &link );
|
|
|
|
IDispatchMsg( pMsg, playerID );
|
|
|
|
}
|
|
|
|
|
|
|
|
void plNetLinkingMgr::LinkToMyNeighborhoodAge( UInt32 playerID )
|
|
|
|
{
|
|
|
|
if ( !fLinkingEnabled )
|
|
|
|
{
|
|
|
|
hsLogEntry( plNetClientMgr::GetInstance()->DebugMsg( "Not linking. Linking is disabled." ) );
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
plNetClientMgr * nc = plNetClientMgr::GetInstance();
|
|
|
|
|
|
|
|
plAgeLinkStruct link;
|
|
|
|
if (!VaultGetLinkToMyNeighborhood(&link))
|
|
|
|
return;
|
|
|
|
|
|
|
|
link.SetLinkingRules( plNetCommon::LinkingRules::kOwnedBook );
|
|
|
|
|
|
|
|
plLinkToAgeMsg* pMsg = TRACKED_NEW plLinkToAgeMsg( &link );
|
|
|
|
IDispatchMsg( pMsg, playerID );
|
|
|
|
}
|
|
|
|
|
|
|
|
void plNetLinkingMgr::LinkPlayerHere( UInt32 playerID )
|
|
|
|
{
|
|
|
|
if ( !fLinkingEnabled )
|
|
|
|
{
|
|
|
|
hsLogEntry( plNetClientMgr::GetInstance()->DebugMsg( "Not linking. Linking is disabled." ) );
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
plNetClientMgr * nc = plNetClientMgr::GetInstance();
|
|
|
|
|
|
|
|
// send the player our current age info so they can link here.
|
|
|
|
plAgeLinkStruct link;
|
|
|
|
link.GetAgeInfo()->CopyFrom( GetAgeLink()->GetAgeInfo() );
|
|
|
|
LinkPlayerToAge( &link, playerID );
|
|
|
|
}
|
|
|
|
|
|
|
|
void plNetLinkingMgr::LinkPlayerToAge( plAgeLinkStruct * link, UInt32 playerID )
|
|
|
|
{
|
|
|
|
if ( !fLinkingEnabled )
|
|
|
|
{
|
|
|
|
hsLogEntry( plNetClientMgr::GetInstance()->DebugMsg( "Not linking. Linking is disabled." ) );
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
plNetClientMgr * nc = plNetClientMgr::GetInstance();
|
|
|
|
|
|
|
|
// send the player the age link so they can link there.
|
|
|
|
link->SetLinkingRules( plNetCommon::LinkingRules::kBasicLink );
|
|
|
|
plLinkToAgeMsg* pMsg = TRACKED_NEW plLinkToAgeMsg( link );
|
|
|
|
IDispatchMsg( pMsg, playerID );
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// link the player back to his previous age
|
|
|
|
//
|
|
|
|
void plNetLinkingMgr::LinkPlayerToPrevAge( UInt32 playerID )
|
|
|
|
{
|
|
|
|
if ( !fLinkingEnabled )
|
|
|
|
{
|
|
|
|
hsLogEntry( plNetClientMgr::GetInstance()->DebugMsg( "Not linking. Linking is disabled." ) );
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Send the player a msg telling them to link to their last age
|
|
|
|
plNetClientMgr * nc = plNetClientMgr::GetInstance();
|
|
|
|
|
|
|
|
plLinkingMgrMsg* pMsg = TRACKED_NEW plLinkingMgrMsg();
|
|
|
|
pMsg->SetCmd( kLinkPlayerToPrevAge);
|
|
|
|
IDispatchMsg( pMsg, playerID );
|
|
|
|
}
|
|
|
|
|
|
|
|
void plNetLinkingMgr::LinkToPlayersAge( UInt32 playerID )
|
|
|
|
{
|
|
|
|
if ( !fLinkingEnabled )
|
|
|
|
{
|
|
|
|
hsLogEntry( plNetClientMgr::GetInstance()->DebugMsg( "Not linking. Linking is disabled." ) );
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// Send the player a msg telling them to send us a msg to link to them. isn't that fun? :)
|
|
|
|
plNetClientMgr * nc = plNetClientMgr::GetInstance();
|
|
|
|
|
|
|
|
plLinkingMgrMsg* pMsg = TRACKED_NEW plLinkingMgrMsg();
|
|
|
|
pMsg->SetCmd( kLinkPlayerHere );
|
|
|
|
pMsg->GetArgs()->AddInt( 0, NetCommGetPlayer()->playerInt ); // send them our id.
|
|
|
|
IDispatchMsg( pMsg, playerID );
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
|
|
|
void plNetLinkingMgr::OfferLinkToPlayer( const plAgeLinkStruct * inInfo, UInt32 playerID )
|
|
|
|
{
|
|
|
|
|
|
|
|
plNetClientMgr *mgr = plNetClientMgr::GetInstance();
|
|
|
|
plLinkToAgeMsg * linkM = TRACKED_NEW plLinkToAgeMsg(inInfo);
|
|
|
|
linkM->AddReceiver(mgr->GetKey());
|
|
|
|
|
|
|
|
plKey host = mgr->GetLocalPlayerKey();
|
|
|
|
|
|
|
|
plNetTransport &transport = mgr->TransportMgr();
|
|
|
|
int guestIdx = transport.FindMember(playerID);
|
|
|
|
plNetTransportMember *guestMem = transport.GetMember(guestIdx); // -1 ?
|
|
|
|
if(guestMem)
|
|
|
|
{
|
|
|
|
plKey guest = guestMem->GetAvatarKey();
|
|
|
|
plAvatarMgr::OfferLinkingBook(host, guest, linkM, host);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// my special version - cjp
|
|
|
|
void plNetLinkingMgr::OfferLinkToPlayer( const plAgeLinkStruct * inInfo, UInt32 playerID, plKey replyKey )
|
|
|
|
{
|
|
|
|
|
|
|
|
plNetClientMgr *mgr = plNetClientMgr::GetInstance();
|
|
|
|
plLinkToAgeMsg * linkM = TRACKED_NEW plLinkToAgeMsg(inInfo);
|
|
|
|
linkM->AddReceiver(mgr->GetKey());
|
|
|
|
|
|
|
|
plKey host = mgr->GetLocalPlayerKey();
|
|
|
|
|
|
|
|
plNetTransport &transport = mgr->TransportMgr();
|
|
|
|
int guestIdx = transport.FindMember(playerID);
|
|
|
|
plNetTransportMember *guestMem = transport.GetMember(guestIdx); // -1 ?
|
|
|
|
if(guestMem)
|
|
|
|
{
|
|
|
|
plKey guest = guestMem->GetAvatarKey();
|
|
|
|
plAvatarMgr::OfferLinkingBook(host, guest, linkM, replyKey);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// for backwards compatibility
|
|
|
|
void plNetLinkingMgr::OfferLinkToPlayer( const plAgeInfoStruct * inInfo, UInt32 playerID )
|
|
|
|
{
|
|
|
|
plAgeLinkStruct *ageLink = TRACKED_NEW plAgeLinkStruct;
|
|
|
|
|
|
|
|
ageLink->GetAgeInfo()->CopyFrom(inInfo);
|
|
|
|
ageLink->SetLinkingRules(plNetCommon::LinkingRules::kBasicLink);
|
|
|
|
OfferLinkToPlayer(ageLink, playerID);
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void plNetLinkingMgr::IPostProcessLink( void )
|
|
|
|
{
|
|
|
|
// Grab some useful things...
|
|
|
|
plAgeLinkStruct* link = GetAgeLink();
|
|
|
|
plAgeInfoStruct* info = link->GetAgeInfo();
|
|
|
|
|
|
|
|
bool city = (stricmp(info->GetAgeFilename(), kCityAgeFilename) == 0);
|
|
|
|
bool hood = (stricmp(info->GetAgeFilename(), kNeighborhoodAgeFilename) == 0);
|
|
|
|
bool psnl = (stricmp(info->GetAgeFilename(), kPersonalAgeFilename) == 0);
|
|
|
|
|
|
|
|
// Update our online status
|
|
|
|
if (RelVaultNode* rvnInfo = VaultGetPlayerInfoNodeIncRef()) {
|
|
|
|
VaultPlayerInfoNode accInfo(rvnInfo);
|
|
|
|
wchar ageInstName[MAX_PATH];
|
|
|
|
Uuid ageInstGuid = *info->GetAgeInstanceGuid();
|
|
|
|
StrToUnicode(ageInstName, info->GetAgeInstanceName(), arrsize(ageInstName));
|
|
|
|
accInfo.SetAgeInstName(ageInstName);
|
|
|
|
accInfo.SetAgeInstUuid(ageInstGuid);
|
|
|
|
accInfo.SetOnline(true);
|
|
|
|
rvnInfo->DecRef();
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (link->GetLinkingRules()) {
|
|
|
|
|
|
|
|
case plNetCommon::LinkingRules::kOwnedBook: {
|
|
|
|
// SPECIAL CASE: City: Every player ever created would be in the list; avoid that.
|
|
|
|
if (city)
|
|
|
|
break;
|
|
|
|
|
|
|
|
{ // Ensure we're in the AgeOwners folder
|
|
|
|
RelVaultNode* fldr = VaultGetAgeAgeOwnersFolderIncRef();
|
|
|
|
RelVaultNode* info = VaultGetPlayerInfoNodeIncRef();
|
|
|
|
|
|
|
|
if (fldr && info)
|
|
|
|
if (!fldr->IsParentOf(info->nodeId, 1))
|
|
|
|
VaultAddChildNode(
|
|
|
|
fldr->nodeId,
|
|
|
|
info->nodeId,
|
|
|
|
NetCommGetPlayer()->playerInt,
|
|
|
|
nil,
|
|
|
|
nil
|
|
|
|
);
|
|
|
|
|
|
|
|
if (fldr)
|
|
|
|
fldr->DecRef();
|
|
|
|
if (info)
|
|
|
|
info->DecRef();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case plNetCommon::LinkingRules::kVisitBook: {
|
|
|
|
// SPECIAL CASE: City: Every player ever created would be in the list; avoid that.
|
|
|
|
if (city)
|
|
|
|
break;
|
|
|
|
|
|
|
|
{ // Ensure we're in the CanVisit folder
|
|
|
|
RelVaultNode* fldr = VaultGetAgeCanVisitFolderIncRef();
|
|
|
|
RelVaultNode* info = VaultGetPlayerInfoNodeIncRef();
|
|
|
|
|
|
|
|
if (fldr && info)
|
|
|
|
if (!fldr->IsParentOf(info->nodeId, 1))
|
|
|
|
VaultAddChildNode(
|
|
|
|
fldr->nodeId,
|
|
|
|
info->nodeId,
|
|
|
|
NetCommGetPlayer()->playerInt,
|
|
|
|
nil,
|
|
|
|
nil
|
|
|
|
);
|
|
|
|
|
|
|
|
if (fldr)
|
|
|
|
fldr->DecRef();
|
|
|
|
if (info)
|
|
|
|
info->DecRef();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case plNetCommon::LinkingRules::kSubAgeBook: {
|
|
|
|
// Register the previous age as a sub age of the current one so that we can link back to that instance
|
|
|
|
plAgeLinkStruct subAgeLink;
|
|
|
|
VaultAgeFindOrCreateSubAgeLink(GetPrevAgeLink()->GetAgeInfo(), &subAgeLink, NetCommGetAge()->ageInstId);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
UInt8 plNetLinkingMgr::IPreProcessLink(void)
|
|
|
|
{
|
|
|
|
// Grab some stuff we're gonna use extensively
|
|
|
|
plNetClientMgr* nc = plNetClientMgr::GetInstance();
|
|
|
|
plAgeLinkStruct* link = GetAgeLink();
|
|
|
|
plAgeInfoStruct* info = link->GetAgeInfo();
|
|
|
|
|
|
|
|
PreProcessResult success = kLinkImmediately;
|
|
|
|
|
|
|
|
if ( nc->GetFlagsBit( plNetClientMgr::kNullSend ) )
|
|
|
|
{
|
|
|
|
hsLogEntry( nc->DebugMsg( "NetClientMgr nullsend. Not linking." ) );
|
|
|
|
return kLinkFailed;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
// Appear offline until we're done linking
|
|
|
|
if (RelVaultNode * rvnInfo = VaultGetPlayerInfoNodeIncRef()) {
|
|
|
|
VaultPlayerInfoNode accInfo(rvnInfo);
|
|
|
|
accInfo.SetAgeInstName(nil);
|
|
|
|
accInfo.SetAgeInstUuid(kNilGuid);
|
|
|
|
accInfo.SetOnline(false);
|
|
|
|
rvnInfo->DecRef();
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
// Update our online status
|
|
|
|
if (RelVaultNode * rvnInfo = VaultGetPlayerInfoNodeIncRef()) {
|
|
|
|
VaultPlayerInfoNode accInfo(rvnInfo);
|
|
|
|
wchar ageInstName[MAX_PATH];
|
|
|
|
Uuid ageInstGuid = *GetAgeLink()->GetAgeInfo()->GetAgeInstanceGuid();
|
|
|
|
StrToUnicode(ageInstName, info->GetAgeInstanceName(), arrsize(ageInstName));
|
|
|
|
accInfo.SetAgeInstName(ageInstName);
|
|
|
|
accInfo.SetAgeInstUuid(ageInstGuid);
|
|
|
|
accInfo.SetOnline(true);
|
|
|
|
rvnInfo->DecRef();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
//------------------------------------------------------------------------
|
|
|
|
// Fixup empty fields
|
|
|
|
if (info->HasAgeFilename())
|
|
|
|
{
|
|
|
|
info->SetAgeFilename(plNetLinkingMgr::GetProperAgeName(info->GetAgeFilename()).c_str());
|
|
|
|
|
|
|
|
if (!info->HasAgeInstanceName())
|
|
|
|
{
|
|
|
|
info->SetAgeInstanceName(info->GetAgeFilename());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
hsLogEntry(nc->DebugMsg( "plNetLinkingMgr: Pre-Process: Linking with %s rules...",
|
|
|
|
plNetCommon::LinkingRules::LinkingRuleStr(link->GetLinkingRules())));
|
|
|
|
|
|
|
|
//------------------------------------------------------------------------
|
|
|
|
// SPECIAL CASE: StartUp: force basic link
|
|
|
|
if (stricmp(info->GetAgeFilename(), kStartUpAgeFilename) == 0)
|
|
|
|
link->SetLinkingRules( plNetCommon::LinkingRules::kBasicLink );
|
|
|
|
|
|
|
|
//------------------------------------------------------------------------
|
|
|
|
// SPECIAL CASE: Nexus: force original link
|
|
|
|
if (stricmp(info->GetAgeFilename(), kNexusAgeFilename) == 0)
|
|
|
|
link->SetLinkingRules(plNetCommon::LinkingRules::kOriginalBook);
|
|
|
|
|
|
|
|
//------------------------------------------------------------------------
|
|
|
|
// SPECIAL CASE: ACA: force original link
|
|
|
|
if (stricmp(info->GetAgeFilename(), kAvCustomizationFilename ) == 0)
|
|
|
|
link->SetLinkingRules(plNetCommon::LinkingRules::kOriginalBook);
|
|
|
|
|
|
|
|
hsLogEntry(nc->DebugMsg("plNetLinkingMgr: Process: Linking with %s rules...",
|
|
|
|
plNetCommon::LinkingRules::LinkingRuleStr(link->GetLinkingRules())));
|
|
|
|
|
|
|
|
switch (link->GetLinkingRules())
|
|
|
|
{
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
// BASIC LINK. Link to a unique instance of the age, if no instance specified.
|
|
|
|
case plNetCommon::LinkingRules::kBasicLink:
|
|
|
|
if (!info->HasAgeInstanceGuid())
|
|
|
|
info->SetAgeInstanceGuid(&plUUID(GuidGenerate()));
|
|
|
|
break;
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
// ORIGINAL BOOK. Become an owner of the age, if not already one.
|
|
|
|
case plNetCommon::LinkingRules::kOriginalBook:
|
|
|
|
{
|
|
|
|
// create a new ageinfo struct with the filename of the age because
|
|
|
|
// we just want to find out if we own *any* link to the age, not the specific
|
|
|
|
// link that we're linking through
|
|
|
|
plAgeInfoStruct ageInfo;
|
|
|
|
ageInfo.SetAgeFilename(info->GetAgeFilename());
|
|
|
|
|
|
|
|
plAgeLinkStruct ownedLink;
|
|
|
|
if (!VaultGetOwnedAgeLink(&ageInfo, &ownedLink)) {
|
|
|
|
// Fill in fields for new age create.
|
|
|
|
if (!info->HasAgeUserDefinedName())
|
|
|
|
{
|
|
|
|
// set user-defined name
|
|
|
|
std::string title;
|
|
|
|
unsigned nameLen = StrLen(nc->GetPlayerName());
|
|
|
|
if (nc->GetPlayerName()[nameLen - 1] == 's' || nc->GetPlayerName()[nameLen - 1] == 'S')
|
|
|
|
xtl::format(title, "%s'", nc->GetPlayerName());
|
|
|
|
else
|
|
|
|
xtl::format(title, "%s's", nc->GetPlayerName());
|
|
|
|
info->SetAgeUserDefinedName(title.c_str());
|
|
|
|
}
|
|
|
|
if (!info->HasAgeDescription())
|
|
|
|
{
|
|
|
|
// set description
|
|
|
|
std::string desc;
|
|
|
|
unsigned nameLen = StrLen(nc->GetPlayerName());
|
|
|
|
if (nc->GetPlayerName()[nameLen - 1] == 's' || nc->GetPlayerName()[nameLen - 1] == 'S')
|
|
|
|
xtl::format(desc, "%s' %s", nc->GetPlayerName(), info->GetAgeInstanceName());
|
|
|
|
else
|
|
|
|
xtl::format(desc, "%s's %s", nc->GetPlayerName(), info->GetAgeInstanceName());
|
|
|
|
info->SetAgeDescription(desc.c_str());
|
|
|
|
}
|
|
|
|
if (!info->HasAgeInstanceGuid()) {
|
|
|
|
info->SetAgeInstanceGuid(&plUUID(GuidGenerate()));
|
|
|
|
}
|
|
|
|
|
|
|
|
// register this as an owned age now before we link to it.
|
|
|
|
// Note: We MUST break or the OwnedBook code will fail!
|
|
|
|
VaultRegisterOwnedAge(link);
|
|
|
|
success = kLinkDeferred;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else if (RelVaultNode* linkNode = VaultGetOwnedAgeLinkIncRef(&ageInfo)) {
|
|
|
|
// We have the age in our AgesIOwnFolder. If its volatile, dump it for the new one.
|
|
|
|
VaultAgeLinkNode linkAcc(linkNode);
|
|
|
|
if (linkAcc.volat) {
|
|
|
|
if (VaultUnregisterOwnedAgeAndWait(&ageInfo)) {
|
|
|
|
// Fill in fields for new age create.
|
|
|
|
if (!info->HasAgeUserDefinedName())
|
|
|
|
{
|
|
|
|
// set user-defined name
|
|
|
|
std::string title;
|
|
|
|
unsigned nameLen = StrLen(nc->GetPlayerName());
|
|
|
|
if (nc->GetPlayerName()[nameLen - 1] == 's' || nc->GetPlayerName()[nameLen - 1] == 'S')
|
|
|
|
xtl::format(title, "%s'", nc->GetPlayerName());
|
|
|
|
else
|
|
|
|
xtl::format(title, "%s's", nc->GetPlayerName());
|
|
|
|
info->SetAgeUserDefinedName(title.c_str());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!info->HasAgeDescription())
|
|
|
|
{
|
|
|
|
// set description
|
|
|
|
std::string desc;
|
|
|
|
unsigned nameLen = StrLen(nc->GetPlayerName());
|
|
|
|
if (nc->GetPlayerName()[nameLen - 1] == 's' || nc->GetPlayerName()[nameLen - 1] == 'S')
|
|
|
|
xtl::format(desc, "%s' %s", nc->GetPlayerName(), info->GetAgeInstanceName());
|
|
|
|
else
|
|
|
|
xtl::format(desc, "%s's %s", nc->GetPlayerName(), info->GetAgeInstanceName());
|
|
|
|
info->SetAgeDescription( desc.c_str() );
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!info->HasAgeInstanceGuid()) {
|
|
|
|
info->SetAgeInstanceGuid(&plUUID(GuidGenerate()));
|
|
|
|
}
|
|
|
|
|
|
|
|
VaultRegisterOwnedAge(link);
|
|
|
|
|
|
|
|
// Note: We MUST break or the OwnedBook code will fail!
|
|
|
|
success = kLinkDeferred;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (stricmp(info->GetAgeFilename(), kNeighborhoodAgeFilename) == 0) {
|
|
|
|
// if we get here then it's because we're linking to a neighborhood that we don't belong to
|
|
|
|
// and our own neighborhood book is not volatile, so really we want to basic link
|
|
|
|
link->SetLinkingRules(plNetCommon::LinkingRules::kBasicLink);
|
|
|
|
success = kLinkImmediately;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
linkNode->DecRef();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
link->SetLinkingRules(plNetCommon::LinkingRules::kOwnedBook);
|
|
|
|
// falls thru to OWNED BOOK case...
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
// OWNED BOOK. Look for the book in our AgesIOwn folder
|
|
|
|
case plNetCommon::LinkingRules::kOwnedBook:
|
|
|
|
{
|
|
|
|
plAgeLinkStruct ownedLink;
|
|
|
|
if (VaultGetOwnedAgeLink(info, &ownedLink)) {
|
|
|
|
info->CopyFrom(ownedLink.GetAgeInfo());
|
|
|
|
// Remember spawn point (treasure book support)
|
|
|
|
plSpawnPointInfo theSpawnPt = link->SpawnPoint();
|
|
|
|
VaultAddOwnedAgeSpawnPoint(*info->GetAgeInstanceGuid(), theSpawnPt);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
success = kLinkFailed;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
// VISIT BOOK. Look for the book in our AgesICanVisit folder
|
|
|
|
case plNetCommon::LinkingRules::kVisitBook:
|
|
|
|
{
|
|
|
|
plAgeLinkStruct visitLink;
|
|
|
|
if (VaultGetVisitAgeLink(info, &visitLink))
|
|
|
|
info->CopyFrom(visitLink.GetAgeInfo());
|
|
|
|
else
|
|
|
|
success = kLinkFailed;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
// SUB-AGE BOOK: Look for an existing sub-age in this age's SubAges folder and use it if found.
|
|
|
|
// plNetClientTaskHandler will add a SubAge back link to our current age once we arrive there.
|
|
|
|
case plNetCommon::LinkingRules::kSubAgeBook:
|
|
|
|
{
|
|
|
|
plAgeLinkStruct subAgeLink;
|
|
|
|
if (VaultAgeFindOrCreateSubAgeLink(info, &subAgeLink, NetCommGetAge()->ageInstId))
|
|
|
|
info->CopyFrom(subAgeLink.GetAgeInfo());
|
|
|
|
else
|
|
|
|
success = kLinkDeferred;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
// CHILD-AGE BOOK: Look for an existing child-age in this parent ageinfo ChildAges folder and use it if found.
|
|
|
|
// plNetClientTaskHandler will add a ChildAge back link to our current age once we arrive there.
|
|
|
|
case plNetCommon::LinkingRules::kChildAgeBook:
|
|
|
|
{
|
|
|
|
plAgeLinkStruct childLink;
|
|
|
|
wchar parentAgeName[MAX_PATH];
|
|
|
|
if (link->HasParentAgeFilename())
|
|
|
|
StrToUnicode(parentAgeName, link->GetParentAgeFilename(), arrsize(parentAgeName));
|
|
|
|
|
|
|
|
switch(VaultAgeFindOrCreateChildAgeLink(
|
|
|
|
(link->HasParentAgeFilename() ? parentAgeName : nil),
|
|
|
|
info,
|
|
|
|
&childLink))
|
|
|
|
{
|
|
|
|
case hsFail:
|
|
|
|
success = kLinkFailed;
|
|
|
|
break;
|
|
|
|
case FALSE:
|
|
|
|
success = kLinkDeferred;
|
|
|
|
break;
|
|
|
|
case TRUE:
|
|
|
|
success = kLinkImmediately;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (success == kLinkImmediately)
|
|
|
|
info->CopyFrom(childLink.GetAgeInfo());
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
// ???
|
|
|
|
DEFAULT_FATAL(link->GetLinkingRules());
|
|
|
|
}
|
|
|
|
|
|
|
|
hsLogEntry(nc->DebugMsg( "plNetLinkingMgr: Post-Process: Linking with %s rules...",
|
|
|
|
plNetCommon::LinkingRules::LinkingRuleStr(link->GetLinkingRules())));
|
|
|
|
|
|
|
|
hsAssert(info->HasAgeFilename(), "AgeLink has no AgeFilename. Link will fail.");
|
|
|
|
|
|
|
|
return success;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
void plNetLinkingMgr::LeaveAge (bool quitting) {
|
|
|
|
// Queue leave op
|
|
|
|
NlmLeaveAgeOp * leaveAgeOp = NEWZERO(NlmLeaveAgeOp);
|
|
|
|
leaveAgeOp->quitting = quitting;
|
|
|
|
QueueOp(leaveAgeOp);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// End.
|