You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
640 lines
19 KiB
640 lines
19 KiB
/*==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==*/ |
|
/***************************************************************************** |
|
* |
|
* $/Plasma20/Sources/Plasma/PubUtilLib/plVault/plVaultNodeAccess.cpp |
|
* |
|
***/ |
|
|
|
#include "Pch.h" |
|
#pragma hdrstop |
|
|
|
//============================================================================ |
|
// Volatile Vault Node Fields - be very careful when adding to this |
|
//============================================================================ |
|
struct NodeTypeToVolatileField { |
|
unsigned nodeType; |
|
uint64_t volatileFields; |
|
}; |
|
|
|
NodeTypeToVolatileField volatileFieldList[] = { |
|
{plVault::kNodeType_PlayerInfo, VaultPlayerInfoNode::kOnline | VaultPlayerInfoNode::kAgeInstName | VaultPlayerInfoNode::kAgeInstUuid}, |
|
{0, 0} |
|
}; |
|
|
|
//============================================================================ |
|
uint64_t GetNodeVolatileFields(NetVaultNode* node) { |
|
uint64_t volatileFields = 0; |
|
unsigned index = 0; |
|
|
|
while (volatileFieldList[index].nodeType != 0) { |
|
if (node->GetNodeType() == volatileFieldList[index].nodeType) { |
|
volatileFields |= volatileFieldList[index].volatileFields; |
|
break; |
|
} |
|
|
|
++index; |
|
} |
|
|
|
return volatileFields; |
|
} |
|
|
|
|
|
/***************************************************************************** |
|
* |
|
* VaultTextNoteNode |
|
* |
|
***/ |
|
|
|
//============================================================================ |
|
enum EAgeInfoFields { |
|
kAgeFilename, |
|
kAgeInstName, |
|
kAgeUserName, |
|
kAgeDesc, |
|
kAgeInstGuid, |
|
kAgeLanguage, |
|
kAgeSequence, |
|
kNumAgeInfoFields |
|
}; |
|
|
|
#ifdef CLIENT |
|
void VaultTextNoteNode::SetVisitInfo (const plAgeInfoStruct & info) { |
|
|
|
ARRAY(wchar_t) buf; |
|
|
|
for (unsigned i = 0; i < kNumAgeInfoFields; ++i) { |
|
switch (i) { |
|
case kAgeFilename: { |
|
wchar_t src[128]; |
|
StrToUnicode(src, info.GetAgeFilename(), arrsize(src)); |
|
unsigned len = StrLen(src); |
|
wchar_t * dst = buf.New(len); |
|
memcpy(dst, src, len * sizeof(src[0])); |
|
} |
|
break; |
|
|
|
case kAgeInstName: { |
|
wchar_t src[128]; |
|
StrToUnicode(src, info.GetAgeInstanceName(), arrsize(src)); |
|
unsigned len = StrLen(src); |
|
wchar_t * dst = buf.New(len); |
|
memcpy(dst, src, len * sizeof(src[0])); |
|
} |
|
break; |
|
|
|
case kAgeUserName: { |
|
wchar_t src[128]; |
|
StrToUnicode(src, info.GetAgeUserDefinedName(), arrsize(src)); |
|
unsigned len = StrLen(src); |
|
wchar_t * dst = buf.New(len); |
|
memcpy(dst, src, len * sizeof(src[0])); |
|
} |
|
break; |
|
|
|
case kAgeDesc: { |
|
wchar_t src[128]; |
|
StrToUnicode(src, info.GetAgeDescription(), arrsize(src)); |
|
unsigned len = StrLen(src); |
|
wchar_t * dst = buf.New(len); |
|
memcpy(dst, src, len * sizeof(src[0])); |
|
} |
|
break; |
|
|
|
case kAgeInstGuid: { |
|
plUUID guid = *info.GetAgeInstanceGuid(); |
|
wchar_t src[64]; |
|
wcsncpy(src, guid.AsString().ToWchar(), 64); |
|
unsigned len = StrLen(src); |
|
wchar_t * dst = buf.New(len); |
|
memcpy(dst, src, len * sizeof(src[0])); |
|
} |
|
break; |
|
|
|
case kAgeLanguage: { |
|
wchar_t src[32]; |
|
StrPrintf(src, arrsize(src), L"%u", info.GetAgeLanguage()); |
|
unsigned len = StrLen(src); |
|
wchar_t * dst = buf.New(len); |
|
memcpy(dst, src, len * sizeof(src[0])); |
|
} |
|
break; |
|
|
|
case kAgeSequence: { |
|
wchar_t src[32]; |
|
StrPrintf(src, arrsize(src), L"%u", info.GetAgeSequenceNumber()); |
|
unsigned len = StrLen(src); |
|
wchar_t * dst = buf.New(len); |
|
memcpy(dst, src, len * sizeof(src[0])); |
|
} |
|
break; |
|
|
|
DEFAULT_FATAL(i); |
|
} |
|
|
|
wchar_t * sep = buf.New(1); |
|
*sep = L'|'; |
|
} |
|
|
|
wchar_t * term = buf.New(1); |
|
*term = 0; |
|
|
|
SetNoteText(buf.Ptr()); |
|
} |
|
#endif |
|
|
|
//============================================================================ |
|
#ifdef CLIENT |
|
bool VaultTextNoteNode::GetVisitInfo (plAgeInfoStruct * info) { |
|
|
|
wchar_t * mem; |
|
const wchar_t * str = mem = wcsdup(GetNoteText()); |
|
|
|
for (unsigned i = 0; i < kNumAgeInfoFields; ++i) { |
|
|
|
wchar_t token[1024]; |
|
switch (i) { |
|
case kAgeFilename: { |
|
StrTokenize(&str, token, arrsize(token), L"|", 1); |
|
if (StrLen(token) > 0) { |
|
char ansi[1024]; |
|
StrToAnsi(ansi, token, arrsize(ansi)); |
|
info->SetAgeFilename(ansi); |
|
} |
|
} |
|
break; |
|
|
|
case kAgeInstName: { |
|
StrTokenize(&str, token, arrsize(token), L"|", 1); |
|
if (StrLen(token) > 0) { |
|
char ansi[1024]; |
|
StrToAnsi(ansi, token, arrsize(ansi)); |
|
info->SetAgeInstanceName(ansi); |
|
} |
|
} |
|
break; |
|
|
|
case kAgeUserName: { |
|
StrTokenize(&str, token, arrsize(token), L"|", 1); |
|
if (StrLen(token) > 0) { |
|
char ansi[1024]; |
|
StrToAnsi(ansi, token, arrsize(ansi)); |
|
info->SetAgeUserDefinedName(ansi); |
|
} |
|
} |
|
break; |
|
|
|
case kAgeDesc: { |
|
StrTokenize(&str, token, arrsize(token), L"|", 1); |
|
if (StrLen(token) > 0) { |
|
char ansi[1024]; |
|
StrToAnsi(ansi, token, arrsize(ansi)); |
|
info->SetAgeDescription(ansi); |
|
} |
|
} |
|
break; |
|
|
|
case kAgeInstGuid: { |
|
StrTokenize(&str, token, arrsize(token), L"|", 1); |
|
if (StrLen(token) > 0) { |
|
plUUID uuid(plString::FromWchar(token)); |
|
info->SetAgeInstanceGuid(&uuid); |
|
} |
|
} |
|
break; |
|
|
|
case kAgeLanguage: { |
|
StrTokenize(&str, token, arrsize(token), L"|", 1); |
|
if (StrLen(token) > 0) { |
|
info->SetAgeLanguage(StrToUnsigned(token, nil, 10)); |
|
} |
|
} |
|
break; |
|
|
|
case kAgeSequence: { |
|
StrTokenize(&str, token, arrsize(token), L"|", 1); |
|
if (StrLen(token) > 0) { |
|
info->SetAgeSequenceNumber(StrToUnsigned(token, nil, 10)); |
|
} |
|
} |
|
break; |
|
|
|
DEFAULT_FATAL(i); |
|
} |
|
} |
|
|
|
free(mem); |
|
return true; |
|
} |
|
#endif |
|
|
|
|
|
/***************************************************************************** |
|
* |
|
* VaultSDLNode |
|
* |
|
***/ |
|
|
|
//============================================================================ |
|
#ifdef CLIENT |
|
bool VaultSDLNode::GetStateDataRecord (plStateDataRecord * rec, unsigned readOptions) { |
|
if (!GetSDLDataLength() || !GetSDLData()) |
|
return false; |
|
|
|
hsRAMStream ram; |
|
ram.Write(GetSDLDataLength(), GetSDLData()); |
|
ram.Rewind(); |
|
|
|
plString sdlRecName; |
|
int sdlRecVersion; |
|
if (!plStateDataRecord::ReadStreamHeader(&ram, &sdlRecName, &sdlRecVersion)) |
|
return false; |
|
|
|
rec->SetDescriptor(sdlRecName, sdlRecVersion); |
|
|
|
// Note: Setting from default here results in a bug causing age SDL to |
|
// be incorrectly shown when immediately linking back to an age you linked |
|
// out of (relto door will be closed, window shut, etc). |
|
// rec->SetFromDefaults(false); |
|
|
|
if (!rec->Read( &ram, 0, readOptions)) |
|
return false; |
|
|
|
// If we converted the record to a newer version, re-save it. |
|
if (rec->GetDescriptor()->GetVersion() != sdlRecVersion) |
|
SetStateDataRecord(rec, readOptions); |
|
|
|
return true; |
|
} |
|
#endif // def CLIENT |
|
|
|
//============================================================================ |
|
#ifdef CLIENT |
|
void VaultSDLNode::SetStateDataRecord (const plStateDataRecord * rec, unsigned writeOptions) { |
|
hsRAMStream ram; |
|
rec->WriteStreamHeader(&ram); |
|
rec->Write(&ram, 0, writeOptions); |
|
ram.Rewind(); |
|
|
|
unsigned bytes = ram.GetEOF(); |
|
uint8_t * buf = nil; |
|
buf = (uint8_t *)malloc(bytes); |
|
|
|
ram.CopyToMem(buf); |
|
SetSDLData(buf, bytes); |
|
free(buf); |
|
} |
|
#endif // def CLIENT |
|
|
|
//============================================================================ |
|
#ifdef CLIENT |
|
void VaultSDLNode::InitStateDataRecord (const wchar_t sdlRecName[], unsigned writeOptions) { |
|
{ |
|
plStateDataRecord * rec = new plStateDataRecord; |
|
bool exists = GetStateDataRecord(rec, 0); |
|
delete rec; |
|
if (exists) |
|
return; |
|
} |
|
|
|
char aStr[MAX_PATH]; |
|
StrToAnsi(aStr, sdlRecName, arrsize(aStr)); |
|
if (plStateDescriptor * des = plSDLMgr::GetInstance()->FindDescriptor(aStr, plSDL::kLatestVersion)) { |
|
plStateDataRecord rec(des); |
|
rec.SetFromDefaults(false); |
|
SetStateDataRecord(&rec, writeOptions|plSDL::kDontWriteDirtyFlag); |
|
} |
|
} |
|
#endif // def CLIENT |
|
|
|
|
|
/***************************************************************************** |
|
* |
|
* VaultImageNode |
|
* |
|
***/ |
|
|
|
//============================================================================ |
|
#ifdef CLIENT |
|
void VaultImageNode::StuffImage (plMipmap * src, int dstType) { |
|
hsRAMStream ramStream; |
|
bool compressSuccess = false; |
|
|
|
switch (dstType) { |
|
case kJPEG: |
|
plJPEG::Instance().SetWriteQuality(70/*percent*/); |
|
compressSuccess = plJPEG::Instance().WriteToStream(&ramStream, src); |
|
break; |
|
case kPNG: |
|
compressSuccess = plPNG::Instance().WriteToStream(&ramStream, src); |
|
break; |
|
default: |
|
break; |
|
} |
|
|
|
if (compressSuccess) { |
|
unsigned bytes = ramStream.GetEOF(); |
|
uint8_t * buffer = (uint8_t *)malloc(bytes); |
|
ramStream.CopyToMem(buffer); |
|
SetImageData(buffer, bytes); |
|
SetImageType(dstType); |
|
free(buffer); |
|
} else { |
|
SetImageData(nil, 0); |
|
SetImageType(kNone); |
|
} |
|
} |
|
#endif |
|
|
|
//============================================================================ |
|
#ifdef CLIENT |
|
bool VaultImageNode::ExtractImage (plMipmap ** dst) { |
|
hsRAMStream ramStream; |
|
ramStream.Write(GetImageDataLength(), GetImageData()); |
|
ramStream.Rewind(); |
|
|
|
switch (GetImageType()) { |
|
case kJPEG: |
|
(*dst) = plJPEG::Instance().ReadFromStream(&ramStream); |
|
break; |
|
|
|
case kPNG: |
|
(*dst) = plPNG::Instance().ReadFromStream(&ramStream); |
|
break; |
|
|
|
case kNone: |
|
default: |
|
(*dst) = nil; |
|
break; |
|
} |
|
return ((*dst) != nil); |
|
} |
|
#endif |
|
|
|
|
|
/***************************************************************************** |
|
* |
|
* VaultAgeLinkNode |
|
* |
|
***/ |
|
|
|
#ifdef CLIENT |
|
struct MatchesSpawnPointTitle |
|
{ |
|
plString fTitle; |
|
MatchesSpawnPointTitle( const plString & title ):fTitle( title ){} |
|
bool operator ()( const plSpawnPointInfo & p ) const { return ( p.fTitle==fTitle ); } |
|
}; |
|
struct MatchesSpawnPointName |
|
{ |
|
plString fName; |
|
MatchesSpawnPointName( const plString & name ):fName( name ){} |
|
bool operator ()( const plSpawnPointInfo & p ) const { return ( p.fSpawnPt==fName ); } |
|
}; |
|
#endif |
|
|
|
//============================================================================ |
|
#ifdef CLIENT |
|
bool VaultAgeLinkNode::CopyTo (plAgeLinkStruct * link) { |
|
if (RelVaultNode * me = VaultGetNodeIncRef(base->GetNodeId())) { |
|
if (RelVaultNode * info = me->GetChildNodeIncRef(plVault::kNodeType_AgeInfo, 1)) { |
|
VaultAgeInfoNode access(info); |
|
access.CopyTo(link->GetAgeInfo()); |
|
me->UnRef(); |
|
return true; |
|
} |
|
me->UnRef(); |
|
} |
|
link->Clear(); |
|
return false; |
|
} |
|
#endif |
|
|
|
//============================================================================ |
|
#ifdef CLIENT |
|
void VaultAgeLinkNode::AddSpawnPoint (const plSpawnPointInfo & point) { |
|
|
|
plSpawnPointVec points; |
|
GetSpawnPoints( &points ); |
|
if ( std::find_if( points.begin(), points.end(), MatchesSpawnPointTitle( point.fTitle ) )!=points.end() ) |
|
return; |
|
|
|
// only check to see if the titles are the same... |
|
//... so we can add the same spawnpoint as long as they have different titles |
|
//if ( std::find_if( points.begin(), points.end(), MatchesSpawnPointName( point.fSpawnPt.c_str() ) )!=points.end() ) |
|
// return; |
|
|
|
points.push_back( point ); |
|
SetSpawnPoints( points ); |
|
} |
|
#endif |
|
|
|
//============================================================================ |
|
#ifdef CLIENT |
|
void VaultAgeLinkNode::RemoveSpawnPoint (const plString & spawnPtName) { |
|
|
|
plSpawnPointVec points; |
|
GetSpawnPoints( &points ); |
|
plSpawnPointVec::iterator it = std::find_if( points.begin(), points.end(), MatchesSpawnPointName( spawnPtName ) ); |
|
while ( it!=points.end() ) |
|
{ |
|
points.erase( it ); |
|
SetSpawnPoints( points ); |
|
it = std::find_if( points.begin(), points.end(), MatchesSpawnPointName( spawnPtName ) ); |
|
} |
|
} |
|
#endif |
|
|
|
//============================================================================ |
|
#ifdef CLIENT |
|
bool VaultAgeLinkNode::HasSpawnPoint (const plString & spawnPtName) const { |
|
|
|
plSpawnPointVec points; |
|
GetSpawnPoints( &points ); |
|
return ( std::find_if( points.begin(), points.end(), MatchesSpawnPointName( spawnPtName ) )!=points.end() ); |
|
} |
|
#endif |
|
|
|
//============================================================================ |
|
#ifdef CLIENT |
|
bool VaultAgeLinkNode::HasSpawnPoint (const plSpawnPointInfo & point) const { |
|
|
|
return HasSpawnPoint(point.GetName()); |
|
} |
|
#endif |
|
|
|
//============================================================================ |
|
#ifdef CLIENT |
|
void VaultAgeLinkNode::GetSpawnPoints (plSpawnPointVec * out) const { |
|
|
|
plString str = plString::FromUtf8(reinterpret_cast<const char*>(GetSpawnPoints()), GetSpawnPointsLength()); |
|
std::vector<plString> izer = str.Tokenize(";"); |
|
for (auto token1 = izer.begin(); token1 != izer.end(); ++token1) |
|
{ |
|
plSpawnPointInfo point; |
|
std::vector<plString> izer2 = token1->Tokenize(":"); |
|
if ( izer2.size() > 0) |
|
point.fTitle = izer2[0]; |
|
if ( izer2.size() > 1) |
|
point.fSpawnPt = izer2[1]; |
|
if ( izer2.size() > 2) |
|
point.fCameraStack = izer2[2]; |
|
|
|
out->push_back(point); |
|
} |
|
} |
|
#endif |
|
|
|
//============================================================================ |
|
#ifdef CLIENT |
|
void VaultAgeLinkNode::SetSpawnPoints (const plSpawnPointVec & in) { |
|
|
|
plStringStream ss; |
|
for ( unsigned i=0; i<in.size(); i++ ) { |
|
ss |
|
<< in[i].fTitle << ":" |
|
<< in[i].fSpawnPt << ":" |
|
<< in[i].fCameraStack << ";"; |
|
} |
|
plString blob = ss.GetString(); |
|
SetSpawnPoints(reinterpret_cast<const uint8_t *>(blob.c_str()), blob.GetSize()); |
|
} |
|
#endif |
|
|
|
/***************************************************************************** |
|
* |
|
* VaultAgeInfoNode |
|
* |
|
***/ |
|
|
|
//============================================================================ |
|
#ifdef CLIENT |
|
const class plUnifiedTime * VaultAgeInfoNode::GetAgeTime () const { |
|
hsAssert(false, "eric, implement me."); |
|
return nil; |
|
} |
|
#endif // def CLIENT |
|
|
|
//============================================================================ |
|
#ifdef CLIENT |
|
void VaultAgeInfoNode::CopyFrom (const plAgeInfoStruct * info) { |
|
wchar_t str[MAX_PATH]; |
|
|
|
// age filename |
|
if (info->HasAgeFilename()) { |
|
StrToUnicode(str, info->GetAgeFilename(), arrsize(str)); |
|
SetAgeFilename(str); |
|
} |
|
else { |
|
SetAgeFilename(nil); |
|
} |
|
|
|
// age instance name |
|
if (info->HasAgeInstanceName()) { |
|
StrToUnicode(str, info->GetAgeInstanceName(), arrsize(str)); |
|
SetAgeInstanceName(str); |
|
} |
|
else { |
|
SetAgeInstanceName(nil); |
|
} |
|
|
|
// age user-defined name |
|
if (info->HasAgeUserDefinedName()) { |
|
StrToUnicode(str, info->GetAgeUserDefinedName(), arrsize(str)); |
|
SetAgeUserDefinedName(str); |
|
} |
|
else { |
|
SetAgeUserDefinedName(nil); |
|
} |
|
|
|
// age description |
|
// TODO |
|
if (info->HasAgeDescription()) { |
|
// StrToUnicode(str, info->GetAgeDescription(), arrsize(str)); |
|
// SetAgeDescription(str); |
|
} |
|
else { |
|
// SetAgeDescription(nil); |
|
} |
|
|
|
// age sequence number |
|
SetAgeSequenceNumber(info->GetAgeSequenceNumber()); |
|
|
|
// age instance guid |
|
SetAgeInstanceGuid(*info->GetAgeInstanceGuid()); |
|
|
|
// age language |
|
SetAgeLanguage(info->GetAgeLanguage()); |
|
} |
|
#endif // def CLIENT |
|
|
|
//============================================================================ |
|
#ifdef CLIENT |
|
void VaultAgeInfoNode::CopyTo (plAgeInfoStruct * info) const { |
|
char str[MAX_PATH]; |
|
|
|
// age filename |
|
StrToAnsi(str, GetAgeFilename(), arrsize(str)); |
|
info->SetAgeFilename(str); |
|
|
|
// age instance name |
|
StrToAnsi(str, GetAgeInstanceName(), arrsize(str)); |
|
info->SetAgeInstanceName(str); |
|
|
|
// age user-defined name |
|
StrToAnsi(str, GetAgeUserDefinedName(), arrsize(str)); |
|
info->SetAgeUserDefinedName(str); |
|
|
|
// age description |
|
// TODO |
|
|
|
// age sequence number |
|
info->SetAgeSequenceNumber(GetAgeSequenceNumber()); |
|
|
|
// age instance guid |
|
plUUID uuid(GetAgeInstanceGuid()); |
|
info->SetAgeInstanceGuid(&uuid); |
|
|
|
// age language |
|
info->SetAgeLanguage(GetAgeLanguage()); |
|
} |
|
#endif // def CLIENT
|
|
|