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.
227 lines
6.9 KiB
227 lines
6.9 KiB
//========= Copyright Valve Corporation, All rights reserved. ============// |
|
// |
|
// Purpose: |
|
// |
|
//============================================================================= |
|
|
|
#include "dmserializers.h" |
|
#include "dmebaseimporter.h" |
|
#include "datamodel/idatamodel.h" |
|
#include "datamodel/dmelement.h" |
|
#include "tier1/KeyValues.h" |
|
#include "tier1/utlbuffer.h" |
|
#include "tier1/utlmap.h" |
|
#include <limits.h> |
|
|
|
|
|
//----------------------------------------------------------------------------- |
|
// Format converter |
|
//----------------------------------------------------------------------------- |
|
class CImportSFMV3 : public CSFMBaseImporter |
|
{ |
|
typedef CSFMBaseImporter BaseClass; |
|
public: |
|
CImportSFMV3( char const *formatName, char const *nextFormatName ); |
|
|
|
private: |
|
virtual bool DoFixup( CDmElement *pSourceRoot ); |
|
|
|
|
|
void FixupElement( CDmElement *pElement ); |
|
// Fixes up all elements |
|
void BuildList( CDmElement *pElement, CUtlRBTree< CDmElement *, int >& list ); |
|
}; |
|
|
|
|
|
//----------------------------------------------------------------------------- |
|
// Singleton instance |
|
//----------------------------------------------------------------------------- |
|
static CImportSFMV3 s_ImportSFMV3( "sfm_v3", "sfm_v4" ); |
|
|
|
void InstallSFMV3Importer( IDataModel *pFactory ) |
|
{ |
|
pFactory->AddLegacyUpdater( &s_ImportSFMV3 ); |
|
} |
|
|
|
|
|
//----------------------------------------------------------------------------- |
|
// Constructor |
|
//----------------------------------------------------------------------------- |
|
CImportSFMV3::CImportSFMV3( char const *formatName, char const *nextFormatName ) : |
|
BaseClass( formatName, nextFormatName ) |
|
{ |
|
} |
|
|
|
|
|
struct LogToCurveInfoTypeMap_t |
|
{ |
|
const char *pLogType; |
|
const char *pLogLayerType; |
|
const char *pCurveInfoType; |
|
}; |
|
|
|
LogToCurveInfoTypeMap_t g_typeMap[] = |
|
{ |
|
{ "DmeIntLog", "DmeIntLogLayer", "DmeIntCurveInfo" }, |
|
{ "DmeFloatLog", "DmeFloatLogLayer", "DmeFloatCurveInfo" }, |
|
{ "DmeBoolLog", "DmeBoolLogLayer", "DmeBoolCurveInfo" }, |
|
// string, |
|
// void, |
|
// objectid, |
|
{ "DmeColorLog", "DmeColorLogLayer", "DmeColorCurveInfo" }, |
|
{ "DmeVector2Log", "DmeVector2LogLayer", "DmeVector2CurveInfo" }, |
|
{ "DmeVector3Log", "DmeVector3LogLayer", "DmeVector3CurveInfo" }, |
|
{ "DmeVector4Log", "DmeVector4LogLayer", "DmeVector4CurveInfo" }, |
|
{ "DmeQAngleLog", "DmeQAngleLogLayer", "DmeQAngleCurveInfo" }, |
|
{ "DmeQuaternionLog", "DmeQuaternionLogLayer","DmeQuaternionCurveInfo" }, |
|
{ "DmeVMatrixLog", "DmeVMatrixLogLayer", "DmeVMatrixCurveInfo" }, |
|
}; |
|
|
|
const char *GetCurveInfoTypeFromLogType( const char *pLogType ) |
|
{ |
|
int c = ARRAYSIZE( g_typeMap ); |
|
for ( int i = 0; i < c; ++i ) |
|
{ |
|
if ( !Q_stricmp( pLogType, g_typeMap[ i ].pLogType ) ) |
|
return g_typeMap[ i ].pCurveInfoType; |
|
} |
|
return NULL; |
|
} |
|
|
|
bool IsLogLayerType( const char *pLogLayerType ) |
|
{ |
|
int c = ARRAYSIZE( g_typeMap ); |
|
for ( int i = 0; i < c; ++i ) |
|
{ |
|
if ( !Q_stricmp( pLogLayerType, g_typeMap[ i ].pLogLayerType ) ) |
|
return true; |
|
} |
|
return false; |
|
} |
|
|
|
void MoveAttribute( CDmElement *pFromElement, const char *pFromAttrName, CDmElement *pToElement = NULL, const char *pToAttrName = NULL, DmAttributeType_t toType = AT_UNKNOWN ) |
|
{ |
|
if ( !pToAttrName ) |
|
{ |
|
pToAttrName = pFromAttrName; |
|
} |
|
|
|
if ( pToElement ) |
|
{ |
|
CDmAttribute *pFromAttr = pFromElement->GetAttribute( pFromAttrName ); |
|
const void *pValue = pFromAttr->GetValueUntyped(); |
|
DmAttributeType_t fromType = pFromAttr->GetType(); |
|
if ( toType == AT_UNKNOWN ) |
|
{ |
|
toType = fromType; |
|
} |
|
|
|
CDmAttribute *pToAttr = pToElement->AddAttribute( pToAttrName, toType ); |
|
if ( !pToAttr ) |
|
{ |
|
Warning( "*** Problem in converter encountered!\n" ); |
|
Warning( "*** Unable to find or add attribute \"%s\" to element \"%s\"!\n", pToAttrName, pToElement->GetName() ); |
|
} |
|
else if ( fromType != toType ) |
|
{ |
|
Warning( "*** Problem in file encountered!\n" ); |
|
Warning( "*** Element \"%s\" has attribute \"%s\" with an unexpected type!\n", pFromElement->GetName(), pFromAttrName ); |
|
} |
|
else |
|
{ |
|
pToAttr->SetValue( toType, pValue ); |
|
} |
|
} |
|
|
|
pFromElement->RemoveAttribute( pFromAttrName ); |
|
} |
|
|
|
// Fixes up all elements |
|
//----------------------------------------------------------------------------- |
|
void CImportSFMV3::FixupElement( CDmElement *pElement ) |
|
{ |
|
if ( !pElement ) |
|
return; |
|
|
|
const char *pType = pElement->GetTypeString(); |
|
|
|
// log layer |
|
if ( IsLogLayerType( pType ) ) |
|
{ |
|
pElement->RemoveAttribute( "ownerlog" ); |
|
return; |
|
} |
|
|
|
// log |
|
const char *pCurveInfoType = GetCurveInfoTypeFromLogType( pType ); |
|
if ( !pCurveInfoType ) |
|
return; |
|
|
|
CDmElement *pCurveInfo = NULL; |
|
CDmAttribute *pUseCurveTypeAttr = pElement->GetAttribute( "usecurvetypes" ); |
|
if ( pUseCurveTypeAttr && pUseCurveTypeAttr->GetValue<bool>() ) |
|
{ |
|
DmElementHandle_t hElement = g_pDataModel->CreateElement( "curve info", pCurveInfoType, pElement->GetFileId() ); |
|
pCurveInfo = g_pDataModel->GetElement( hElement ); |
|
} |
|
pElement->RemoveAttribute( "usecurvetypes" ); |
|
|
|
MoveAttribute( pElement, "defaultcurvetype", pCurveInfo, "defaultCurveType", AT_INT ); |
|
MoveAttribute( pElement, "defaultedgezerovalue",pCurveInfo, "defaultEdgeZeroValue" ); |
|
MoveAttribute( pElement, "useedgeinfo", pCurveInfo, "useEdgeInfo", AT_BOOL ); |
|
MoveAttribute( pElement, "rightedgetime", pCurveInfo, "rightEdgeTime", AT_INT ); |
|
MoveAttribute( pElement, "left_edge_active", pCurveInfo, "leftEdgeActive", AT_BOOL ); |
|
MoveAttribute( pElement, "right_edge_active", pCurveInfo, "rightEdgeActive", AT_BOOL ); |
|
MoveAttribute( pElement, "left_edge_curvetype", pCurveInfo, "leftEdgeCurveType", AT_INT ); |
|
MoveAttribute( pElement, "right_edge_curvetype",pCurveInfo, "rightEdgeCurveType", AT_INT ); |
|
MoveAttribute( pElement, "left_edge_value", pCurveInfo, "leftEdgeValue" ); |
|
MoveAttribute( pElement, "right_edge_value", pCurveInfo, "rightEdgeValue" ); |
|
} |
|
|
|
// Fixes up all elements |
|
//----------------------------------------------------------------------------- |
|
void CImportSFMV3::BuildList( CDmElement *pElement, CUtlRBTree< CDmElement *, int >& list ) |
|
{ |
|
if ( !pElement ) |
|
return; |
|
|
|
if ( list.Find( pElement ) != list.InvalidIndex() ) |
|
return; |
|
|
|
list.Insert( pElement ); |
|
|
|
// Descend to bottom of tree, then do fixup coming back up the tree |
|
for ( CDmAttribute *pAttribute = pElement->FirstAttribute(); pAttribute; pAttribute = pAttribute->NextAttribute() ) |
|
{ |
|
if ( pAttribute->GetType() == AT_ELEMENT ) |
|
{ |
|
CDmElement *pElementAt = pAttribute->GetValueElement<CDmElement>( ); |
|
BuildList( pElementAt, list ); |
|
continue; |
|
} |
|
|
|
if ( pAttribute->GetType() == AT_ELEMENT_ARRAY ) |
|
{ |
|
CDmrElementArray<> array( pAttribute ); |
|
int nCount = array.Count(); |
|
for ( int i = 0; i < nCount; ++i ) |
|
{ |
|
CDmElement *pChild = array[ i ]; |
|
BuildList( pChild, list ); |
|
} |
|
continue; |
|
} |
|
} |
|
} |
|
|
|
bool CImportSFMV3::DoFixup( CDmElement *pSourceRoot ) |
|
{ |
|
CUtlRBTree< CDmElement *, int > fixlist( 0, 0, DefLessFunc( CDmElement * ) ); |
|
BuildList( pSourceRoot, fixlist ); |
|
for ( int i = fixlist.FirstInorder(); i != fixlist.InvalidIndex() ; i = fixlist.NextInorder( i ) ) |
|
{ |
|
// Search and replace in the entire tree! |
|
FixupElement( fixlist[ i ] ); |
|
} |
|
return true; |
|
}
|
|
|