doom3-bfg/neo/framework/DeclAF.cpp

1748 lines
46 KiB
C++
Raw Normal View History

2012-11-26 18:58:24 +00:00
/*
===========================================================================
Doom 3 BFG Edition GPL Source Code
Copyright (C) 1993-2012 id Software LLC, a ZeniMax Media company.
This file is part of the Doom 3 BFG Edition GPL Source Code ("Doom 3 BFG Edition Source Code").
Doom 3 BFG Edition Source Code 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.
Doom 3 BFG Edition Source Code 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 Doom 3 BFG Edition Source Code. If not, see <http://www.gnu.org/licenses/>.
In addition, the Doom 3 BFG Edition Source Code is also subject to certain additional terms. You should have received a copy of these additional terms immediately following the terms and conditions of the GNU General Public License which accompanied the Doom 3 BFG Edition Source Code. If not, please request a copy in writing from id Software at the address below.
If you have questions concerning this license or the applicable additional terms, you may contact in writing id Software LLC, c/o ZeniMax Media Inc., Suite 120, Rockville, Maryland 20850 USA.
===========================================================================
*/
#include "../idlib/precompiled.h"
#pragma hdrstop
/*
===============================================================================
idDeclAF
===============================================================================
*/
/*
================
idAFVector::idAFVector
================
*/
idAFVector::idAFVector() {
type = VEC_COORDS;
vec.Zero();
negate = false;
}
/*
================
idAFVector::Parse
================
*/
bool idAFVector::Parse( idLexer &src ) {
idToken token;
if ( !src.ReadToken( &token ) ) {
return false;
}
if ( token == "-" ) {
negate = true;
if ( !src.ReadToken( &token ) ) {
return false;
}
}
else {
negate = false;
}
if ( token == "(" ) {
type = idAFVector::VEC_COORDS;
vec.x = src.ParseFloat();
src.ExpectTokenString( "," );
vec.y = src.ParseFloat();
src.ExpectTokenString( "," );
vec.z = src.ParseFloat();
src.ExpectTokenString( ")" );
}
else if ( token == "joint" ) {
type = idAFVector::VEC_JOINT;
src.ExpectTokenString( "(" );
src.ReadToken( &token );
joint1 = token;
src.ExpectTokenString( ")" );
}
else if ( token == "bonecenter" ) {
type = idAFVector::VEC_BONECENTER;
src.ExpectTokenString( "(" );
src.ReadToken( &token );
joint1 = token;
src.ExpectTokenString( "," );
src.ReadToken( &token );
joint2 = token;
src.ExpectTokenString( ")" );
}
else if ( token == "bonedir" ) {
type = idAFVector::VEC_BONEDIR;
src.ExpectTokenString( "(" );
src.ReadToken( &token );
joint1 = token;
src.ExpectTokenString( "," );
src.ReadToken( &token );
joint2 = token;
src.ExpectTokenString( ")" );
}
else {
src.Error( "unknown token %s in vector", token.c_str() );
return false;
}
return true;
}
/*
================
idAFVector::Finish
================
*/
bool idAFVector::Finish( const char *fileName, const getJointTransform_t GetJointTransform, const idJointMat *frame, void *model ) const {
idMat3 axis;
idVec3 start, end;
switch( type ) {
case idAFVector::VEC_COORDS: {
break;
}
case idAFVector::VEC_JOINT: {
if ( !GetJointTransform( model, frame, joint1, vec, axis ) ) {
common->Warning( "invalid joint %s in joint() in '%s'", joint1.c_str(), fileName );
vec.Zero();
}
break;
}
case idAFVector::VEC_BONECENTER: {
if ( !GetJointTransform( model, frame, joint1, start, axis ) ) {
common->Warning( "invalid joint %s in bonecenter() in '%s'", joint1.c_str(), fileName );
start.Zero();
}
if ( !GetJointTransform( model, frame, joint2, end, axis ) ) {
common->Warning( "invalid joint %s in bonecenter() in '%s'", joint2.c_str(), fileName );
end.Zero();
}
vec = ( start + end ) * 0.5f;
break;
}
case idAFVector::VEC_BONEDIR: {
if ( !GetJointTransform( model, frame, joint1, start, axis ) ) {
common->Warning( "invalid joint %s in bonedir() in '%s'", joint1.c_str(), fileName );
start.Zero();
}
if ( !GetJointTransform( model, frame, joint2, end, axis ) ) {
common->Warning( "invalid joint %s in bonedir() in '%s'", joint2.c_str(), fileName );
end.Zero();
}
vec = ( end - start );
break;
}
default: {
vec.Zero();
break;
}
}
if ( negate ) {
vec = -vec;
}
return true;
}
/*
================
idAFVector::Write
================
*/
bool idAFVector::Write( idFile *f ) const {
if ( negate ) {
f->WriteFloatString( "-" );
}
switch( type ) {
case idAFVector::VEC_COORDS: {
f->WriteFloatString( "( %f, %f, %f )", vec.x, vec.y, vec.z );
break;
}
case idAFVector::VEC_JOINT: {
f->WriteFloatString( "joint( \"%s\" )", joint1.c_str() );
break;
}
case idAFVector::VEC_BONECENTER: {
f->WriteFloatString( "bonecenter( \"%s\", \"%s\" )", joint1.c_str(), joint2.c_str() );
break;
}
case idAFVector::VEC_BONEDIR: {
f->WriteFloatString( "bonedir( \"%s\", \"%s\" )", joint1.c_str(), joint2.c_str() );
break;
}
default: {
break;
}
}
return true;
}
/*
================
idAFVector::ToString
================
*/
const char *idAFVector::ToString( idStr &str, const int precision ) {
switch( type ) {
case idAFVector::VEC_COORDS: {
char format[128];
sprintf( format, "( %%.%df, %%.%df, %%.%df )", precision, precision, precision );
sprintf( str, format, vec.x, vec.y, vec.z );
break;
}
case idAFVector::VEC_JOINT: {
sprintf( str, "joint( \"%s\" )", joint1.c_str() );
break;
}
case idAFVector::VEC_BONECENTER: {
sprintf( str, "bonecenter( \"%s\", \"%s\" )", joint1.c_str(), joint2.c_str() );
break;
}
case idAFVector::VEC_BONEDIR: {
sprintf( str, "bonedir( \"%s\", \"%s\" )", joint1.c_str(), joint2.c_str() );
break;
}
default: {
break;
}
}
if ( negate ) {
str = "-" + str;
}
return str.c_str();
}
/*
================
idDeclAF_Body::SetDefault
================
*/
void idDeclAF_Body::SetDefault( const idDeclAF *file ) {
name = "noname";
modelType = TRM_BOX;
v1.type = idAFVector::VEC_COORDS;
v1.ToVec3().x = v1.ToVec3().y = v1.ToVec3().z = -10.0f;
v2.type = idAFVector::VEC_COORDS;
v2.ToVec3().x = v2.ToVec3().y = v2.ToVec3().z = 10.0f;
numSides = 3;
origin.ToVec3().Zero();
angles.Zero();
density = 0.2f;
inertiaScale.Identity();
linearFriction = file->defaultLinearFriction;
angularFriction = file->defaultAngularFriction;
contactFriction = file->defaultContactFriction;
contents = file->contents;
clipMask = file->clipMask;
selfCollision = file->selfCollision;
frictionDirection.ToVec3().Zero();
contactMotorDirection.ToVec3().Zero();
jointName = "origin";
jointMod = DECLAF_JOINTMOD_AXIS;
containedJoints = "*origin";
}
/*
================
idDeclAF_Constraint::SetDefault
================
*/
void idDeclAF_Constraint::SetDefault( const idDeclAF *file ) {
name = "noname";
type = DECLAF_CONSTRAINT_UNIVERSALJOINT;
if ( file->bodies.Num() ) {
body1 = file->bodies[0]->name;
}
else {
body1 = "world";
}
body2 = "world";
friction = file->defaultConstraintFriction;
anchor.ToVec3().Zero();
anchor2.ToVec3().Zero();
axis.ToVec3().Set( 1.0f, 0.0f, 0.0f );
shaft[0].ToVec3().Set( 0.0f, 0.0f, -1.0f );
shaft[1].ToVec3().Set( 0.0f, 0.0f, 1.0f );
limit = idDeclAF_Constraint::LIMIT_NONE;
limitAngles[0] =
limitAngles[1] =
limitAngles[2] = 0.0f;
limitAxis.ToVec3().Set( 0.0f, 0.0f, -1.0f );
}
/*
================
idDeclAF::WriteBody
================
*/
bool idDeclAF::WriteBody( idFile *f, const idDeclAF_Body &body ) const {
idStr str;
f->WriteFloatString( "\nbody \"%s\" {\n", body.name.c_str() );
f->WriteFloatString( "\tjoint \"%s\"\n", body.jointName.c_str() );
f->WriteFloatString( "\tmod %s\n", JointModToString( body.jointMod ) );
switch( body.modelType ) {
case TRM_BOX: {
f->WriteFloatString( "\tmodel box( " );
body.v1.Write( f );
f->WriteFloatString( ", " );
body.v2.Write( f );
f->WriteFloatString( " )\n" );
break;
}
case TRM_OCTAHEDRON: {
f->WriteFloatString( "\tmodel octahedron( " );
body.v1.Write( f );
f->WriteFloatString( ", " );
body.v2.Write( f );
f->WriteFloatString( " )\n" );
break;
}
case TRM_DODECAHEDRON: {
f->WriteFloatString( "\tmodel dodecahedron( " );
body.v1.Write( f );
f->WriteFloatString( ", " );
body.v2.Write( f );
f->WriteFloatString( " )\n" );
break;
}
case TRM_CYLINDER: {
f->WriteFloatString( "\tmodel cylinder( " );
body.v1.Write( f );
f->WriteFloatString( ", " );
body.v2.Write( f );
f->WriteFloatString( ", %d )\n", body.numSides );
break;
}
case TRM_CONE: {
f->WriteFloatString( "\tmodel cone( " );
body.v1.Write( f );
f->WriteFloatString( ", " );
body.v2.Write( f );
f->WriteFloatString( ", %d )\n", body.numSides );
break;
}
case TRM_BONE: {
f->WriteFloatString( "\tmodel bone( " );
body.v1.Write( f );
f->WriteFloatString( ", " );
body.v2.Write( f );
f->WriteFloatString( ", %f )\n", body.width );
break;
}
default:
assert( 0 );
break;
}
f->WriteFloatString( "\torigin " );
body.origin.Write( f );
f->WriteFloatString( "\n" );
if ( body.angles != ang_zero ) {
f->WriteFloatString( "\tangles ( %f, %f, %f )\n", body.angles.pitch, body.angles.yaw, body.angles.roll );
}
f->WriteFloatString( "\tdensity %f\n", body.density );
if ( body.inertiaScale != mat3_identity ) {
const idMat3 &ic = body.inertiaScale;
f->WriteFloatString( "\tinertiaScale (%f %f %f %f %f %f %f %f %f)\n",
ic[0][0], ic[0][1], ic[0][2],
ic[1][0], ic[1][1], ic[1][2],
ic[2][0], ic[2][1], ic[2][2] );
}
if ( body.linearFriction != -1 ) {
f->WriteFloatString( "\tfriction %f, %f, %f\n", body.linearFriction, body.angularFriction, body.contactFriction );
}
f->WriteFloatString( "\tcontents %s\n", ContentsToString( body.contents, str ) );
f->WriteFloatString( "\tclipMask %s\n", ContentsToString( body.clipMask, str ) );
f->WriteFloatString( "\tselfCollision %d\n", body.selfCollision );
if ( body.frictionDirection.ToVec3() != vec3_origin ) {
f->WriteFloatString( "\tfrictionDirection " );
body.frictionDirection.Write( f );
f->WriteFloatString( "\n" );
}
if ( body.contactMotorDirection.ToVec3() != vec3_origin ) {
f->WriteFloatString( "\tcontactMotorDirection " );
body.contactMotorDirection.Write( f );
f->WriteFloatString( "\n" );
}
f->WriteFloatString( "\tcontainedJoints \"%s\"\n", body.containedJoints.c_str() );
f->WriteFloatString( "}\n" );
return true;
}
/*
================
idDeclAF::WriteFixed
================
*/
bool idDeclAF::WriteFixed( idFile *f, const idDeclAF_Constraint &c ) const {
f->WriteFloatString( "\nfixed \"%s\" {\n", c.name.c_str() );
f->WriteFloatString( "\tbody1 \"%s\"\n", c.body1.c_str() );
f->WriteFloatString( "\tbody2 \"%s\"\n", c.body2.c_str() );
f->WriteFloatString( "}\n" );
return true;
}
/*
================
idDeclAF::WriteBallAndSocketJoint
================
*/
bool idDeclAF::WriteBallAndSocketJoint( idFile *f, const idDeclAF_Constraint &c ) const {
f->WriteFloatString( "\nballAndSocketJoint \"%s\" {\n", c.name.c_str() );
f->WriteFloatString( "\tbody1 \"%s\"\n", c.body1.c_str() );
f->WriteFloatString( "\tbody2 \"%s\"\n", c.body2.c_str() );
f->WriteFloatString( "\tanchor " );
c.anchor.Write( f );
f->WriteFloatString( "\n" );
f->WriteFloatString( "\tfriction %f\n", c.friction );
if ( c.limit == idDeclAF_Constraint::LIMIT_CONE ) {
f->WriteFloatString( "\tconeLimit " );
c.limitAxis.Write( f );
f->WriteFloatString( ", %f, ", c.limitAngles[0] );
c.shaft[0].Write( f );
f->WriteFloatString( "\n" );
}
else if ( c.limit == idDeclAF_Constraint::LIMIT_PYRAMID ) {
f->WriteFloatString( "\tpyramidLimit " );
c.limitAxis.Write( f );
f->WriteFloatString( ", %f, %f, %f, ", c.limitAngles[0], c.limitAngles[1], c.limitAngles[2] );
c.shaft[0].Write( f );
f->WriteFloatString( "\n" );
}
f->WriteFloatString( "}\n" );
return true;
}
/*
================
idDeclAF::WriteUniversalJoint
================
*/
bool idDeclAF::WriteUniversalJoint( idFile *f, const idDeclAF_Constraint &c ) const {
f->WriteFloatString( "\nuniversalJoint \"%s\" {\n", c.name.c_str() );
f->WriteFloatString( "\tbody1 \"%s\"\n", c.body1.c_str() );
f->WriteFloatString( "\tbody2 \"%s\"\n", c.body2.c_str() );
f->WriteFloatString( "\tanchor " );
c.anchor.Write( f );
f->WriteFloatString( "\n" );
f->WriteFloatString( "\tshafts " );
c.shaft[0].Write( f );
f->WriteFloatString( ", " );
c.shaft[1].Write( f );
f->WriteFloatString( "\n" );
f->WriteFloatString( "\tfriction %f\n", c.friction );
if ( c.limit == idDeclAF_Constraint::LIMIT_CONE ) {
f->WriteFloatString( "\tconeLimit " );
c.limitAxis.Write( f );
f->WriteFloatString( ", %f\n", c.limitAngles[0] );
}
else if ( c.limit == idDeclAF_Constraint::LIMIT_PYRAMID ) {
f->WriteFloatString( "\tpyramidLimit " );
c.limitAxis.Write( f );
f->WriteFloatString( ", %f, %f, %f\n", c.limitAngles[0], c.limitAngles[1], c.limitAngles[2] );
}
f->WriteFloatString( "}\n" );
return true;
}
/*
================
idDeclAF::WriteHinge
================
*/
bool idDeclAF::WriteHinge( idFile *f, const idDeclAF_Constraint &c ) const {
f->WriteFloatString( "\nhinge \"%s\" {\n", c.name.c_str() );
f->WriteFloatString( "\tbody1 \"%s\"\n", c.body1.c_str() );
f->WriteFloatString( "\tbody2 \"%s\"\n", c.body2.c_str() );
f->WriteFloatString( "\tanchor " );
c.anchor.Write( f );
f->WriteFloatString( "\n" );
f->WriteFloatString( "\taxis " );
c.axis.Write( f );
f->WriteFloatString( "\n" );
f->WriteFloatString( "\tfriction %f\n", c.friction );
if ( c.limit == idDeclAF_Constraint::LIMIT_CONE ) {
f->WriteFloatString( "\tlimit " );
f->WriteFloatString( "%f, %f, %f", c.limitAngles[0], c.limitAngles[1], c.limitAngles[2] );
f->WriteFloatString( "\n" );
}
f->WriteFloatString( "}\n" );
return true;
}
/*
================
idDeclAF::WriteSlider
================
*/
bool idDeclAF::WriteSlider( idFile *f, const idDeclAF_Constraint &c ) const {
f->WriteFloatString( "\nslider \"%s\" {\n", c.name.c_str() );
f->WriteFloatString( "\tbody1 \"%s\"\n", c.body1.c_str() );
f->WriteFloatString( "\tbody2 \"%s\"\n", c.body2.c_str() );
f->WriteFloatString( "\taxis " );
c.axis.Write( f );
f->WriteFloatString( "\n" );
f->WriteFloatString( "\tfriction %f\n", c.friction );
f->WriteFloatString( "}\n" );
return true;
}
/*
================
idDeclAF::WriteSpring
================
*/
bool idDeclAF::WriteSpring( idFile *f, const idDeclAF_Constraint &c ) const {
f->WriteFloatString( "\nspring \"%s\" {\n", c.name.c_str() );
f->WriteFloatString( "\tbody1 \"%s\"\n", c.body1.c_str() );
f->WriteFloatString( "\tbody2 \"%s\"\n", c.body2.c_str() );
f->WriteFloatString( "\tanchor1 " );
c.anchor.Write( f );
f->WriteFloatString( "\n" );
f->WriteFloatString( "\tanchor2 " );
c.anchor2.Write( f );
f->WriteFloatString( "\n" );
f->WriteFloatString( "\tfriction %f\n", c.friction );
f->WriteFloatString( "\tstretch %f\n", c.stretch );
f->WriteFloatString( "\tcompress %f\n", c.compress );
f->WriteFloatString( "\tdamping %f\n", c.damping );
f->WriteFloatString( "\trestLength %f\n", c.restLength );
f->WriteFloatString( "\tminLength %f\n", c.minLength );
f->WriteFloatString( "\tmaxLength %f\n", c.maxLength );
f->WriteFloatString( "}\n" );
return true;
}
/*
================
idDeclAF::WriteConstraint
================
*/
bool idDeclAF::WriteConstraint( idFile *f, const idDeclAF_Constraint &c ) const {
switch( c.type ) {
case DECLAF_CONSTRAINT_FIXED:
return WriteFixed( f, c );
case DECLAF_CONSTRAINT_BALLANDSOCKETJOINT:
return WriteBallAndSocketJoint( f, c );
case DECLAF_CONSTRAINT_UNIVERSALJOINT:
return WriteUniversalJoint( f, c );
case DECLAF_CONSTRAINT_HINGE:
return WriteHinge( f, c );
case DECLAF_CONSTRAINT_SLIDER:
return WriteSlider( f, c );
case DECLAF_CONSTRAINT_SPRING:
return WriteSpring( f, c );
default:
break;
}
return false;
}
/*
================
idDeclAF::WriteSettings
================
*/
bool idDeclAF::WriteSettings( idFile *f ) const {
idStr str;
f->WriteFloatString( "\nsettings {\n" );
f->WriteFloatString( "\tmodel \"%s\"\n", model.c_str() );
f->WriteFloatString( "\tskin \"%s\"\n", skin.c_str() );
f->WriteFloatString( "\tfriction %f, %f, %f, %f\n", defaultLinearFriction, defaultAngularFriction, defaultContactFriction, defaultConstraintFriction );
f->WriteFloatString( "\tsuspendSpeed %f, %f, %f, %f\n", suspendVelocity[0], suspendVelocity[1], suspendAcceleration[0], suspendAcceleration[1] );
f->WriteFloatString( "\tnoMoveTime %f\n", noMoveTime );
f->WriteFloatString( "\tnoMoveTranslation %f\n", noMoveTranslation );
f->WriteFloatString( "\tnoMoveRotation %f\n", noMoveRotation );
f->WriteFloatString( "\tminMoveTime %f\n", minMoveTime );
f->WriteFloatString( "\tmaxMoveTime %f\n", maxMoveTime );
f->WriteFloatString( "\ttotalMass %f\n", totalMass );
f->WriteFloatString( "\tcontents %s\n", ContentsToString( contents, str ) );
f->WriteFloatString( "\tclipMask %s\n", ContentsToString( clipMask, str ) );
f->WriteFloatString( "\tselfCollision %d\n", selfCollision );
f->WriteFloatString( "}\n" );
return true;
}
/*
================
idDeclAF::RebuildTextSource
================
*/
bool idDeclAF::RebuildTextSource() {
int i;
idFile_Memory f;
f.WriteFloatString("\n\n/*\n"
"\tGenerated by the Articulated Figure Editor.\n"
"\tDo not edit directly but launch the game and type 'editAFs' on the console.\n"
"*/\n" );
f.WriteFloatString( "\narticulatedFigure %s {\n", GetName() );
if ( !WriteSettings( &f ) ) {
return false;
}
for ( i = 0; i < bodies.Num(); i++ ) {
if ( !WriteBody( &f, *bodies[i] ) ) {
return false;
}
}
for ( i = 0; i < constraints.Num(); i++ ) {
if ( !WriteConstraint( &f, *constraints[i] ) ) {
return false;
}
}
f.WriteFloatString( "\n}" );
SetText( f.GetDataPtr() );
return true;
}
/*
================
idDeclAF::Save
================
*/
bool idDeclAF::Save() {
RebuildTextSource();
ReplaceSourceFileText();
modified = false;
return true;
}
/*
================
idDeclAF::ContentsFromString
================
*/
int idDeclAF::ContentsFromString( const char *str ) {
int c;
idToken token;
idLexer src( str, idStr::Length( str ), "idDeclAF::ContentsFromString" );
c = 0;
while( src.ReadToken( &token ) ) {
if ( token.Icmp( "none" ) == 0 ) {
c = 0;
}
else if ( token.Icmp( "solid" ) == 0 ) {
c |= CONTENTS_SOLID;
}
else if ( token.Icmp( "body" ) == 0 ) {
c |= CONTENTS_BODY;
}
else if ( token.Icmp( "corpse" ) == 0 ) {
c |= CONTENTS_CORPSE;
}
else if ( token.Icmp( "playerclip" ) == 0 ) {
c |= CONTENTS_PLAYERCLIP;
}
else if ( token.Icmp( "monsterclip" ) == 0 ) {
c |= CONTENTS_MONSTERCLIP;
}
else if ( token == "," ) {
continue;
}
else {
return c;
}
}
return c;
}
/*
================
idDeclAF::ContentsToString
================
*/
const char *idDeclAF::ContentsToString( const int contents, idStr &str ) {
str = "";
if ( contents & CONTENTS_SOLID ) {
if ( str.Length() ) str += ", ";
str += "solid";
}
if ( contents & CONTENTS_BODY ) {
if ( str.Length() ) str += ", ";
str += "body";
}
if ( contents & CONTENTS_CORPSE ) {
if ( str.Length() ) str += ", ";
str += "corpse";
}
if ( contents & CONTENTS_PLAYERCLIP ) {
if ( str.Length() ) str += ", ";
str += "playerclip";
}
if ( contents & CONTENTS_MONSTERCLIP ) {
if ( str.Length() ) str += ", ";
str += "monsterclip";
}
if ( str[0] == '\0' ) {
str = "none";
}
return str.c_str();
}
/*
================
idDeclAF::JointModFromString
================
*/
declAFJointMod_t idDeclAF::JointModFromString( const char *str ) {
if ( idStr::Icmp( str, "orientation" ) == 0 ) {
return DECLAF_JOINTMOD_AXIS;
}
if ( idStr::Icmp( str, "position" ) == 0 ) {
return DECLAF_JOINTMOD_ORIGIN;
}
if ( idStr::Icmp( str, "both" ) == 0 ) {
return DECLAF_JOINTMOD_BOTH;
}
return DECLAF_JOINTMOD_AXIS;
}
/*
================
idDeclAF::JointModToString
================
*/
const char * idDeclAF::JointModToString( declAFJointMod_t jointMod ) {
switch( jointMod ) {
case DECLAF_JOINTMOD_AXIS: {
return "orientation";
}
case DECLAF_JOINTMOD_ORIGIN: {
return "position";
}
case DECLAF_JOINTMOD_BOTH: {
return "both";
}
}
return "orientation";
}
/*
=================
idDeclAF::Size
=================
*/
size_t idDeclAF::Size() const {
return sizeof( idDeclAF );
}
/*
================
idDeclAF::ParseContents
================
*/
bool idDeclAF::ParseContents( idLexer &src, int &c ) const {
idToken token;
idStr str;
while( src.ReadToken( &token ) ) {
str += token;
if ( !src.CheckTokenString( "," ) ) {
break;
}
str += ",";
}
c = ContentsFromString( str );
return true;
}
/*
================
idDeclAF::ParseBody
================
*/
bool idDeclAF::ParseBody( idLexer &src ) {
bool hasJoint = false;
idToken token;
idAFVector angles;
idDeclAF_Body *body = new (TAG_DECL) idDeclAF_Body;
bodies.Alloc() = body;
body->SetDefault( this );
if ( !src.ExpectTokenType( TT_STRING, 0, &token ) ||
!src.ExpectTokenString( "{" ) ) {
return false;
}
body->name = token;
if ( !body->name.Icmp( "origin" ) || !body->name.Icmp( "world" ) ) {
src.Error( "a body may not be named \"origin\" or \"world\"" );
return false;
}
while( src.ReadToken( &token ) ) {
if ( !token.Icmp( "model" ) ) {
if ( !src.ExpectTokenType( TT_NAME, 0, &token ) ) {
return false;
}
if ( !token.Icmp( "box" ) ) {
body->modelType = TRM_BOX;
if ( !src.ExpectTokenString( "(" ) ||
!body->v1.Parse( src ) ||
!src.ExpectTokenString( "," ) ||
!body->v2.Parse( src ) ||
!src.ExpectTokenString( ")" ) ) {
return false;
}
} else if ( !token.Icmp( "octahedron" ) ) {
body->modelType = TRM_OCTAHEDRON;
if ( !src.ExpectTokenString( "(" ) ||
!body->v1.Parse( src ) ||
!src.ExpectTokenString( "," ) ||
!body->v2.Parse( src ) ||
!src.ExpectTokenString( ")" ) ) {
return false;
}
} else if ( !token.Icmp( "dodecahedron" ) ) {
body->modelType = TRM_DODECAHEDRON;
if ( !src.ExpectTokenString( "(" ) ||
!body->v1.Parse( src ) ||
!src.ExpectTokenString( "," ) ||
!body->v2.Parse( src ) ||
!src.ExpectTokenString( ")" ) ) {
return false;
}
} else if ( !token.Icmp( "cylinder" ) ) {
body->modelType = TRM_CYLINDER;
if ( !src.ExpectTokenString( "(" ) ||
!body->v1.Parse( src ) ||
!src.ExpectTokenString( "," ) ||
!body->v2.Parse( src ) ||
!src.ExpectTokenString( "," ) ) {
return false;
}
body->numSides = src.ParseInt();
if ( !src.ExpectTokenString( ")" ) ) {
return false;
}
} else if ( !token.Icmp( "cone" ) ) {
body->modelType = TRM_CONE;
if ( !src.ExpectTokenString( "(" ) ||
!body->v1.Parse( src ) ||
!src.ExpectTokenString( "," ) ||
!body->v2.Parse( src ) ||
!src.ExpectTokenString( "," ) ) {
return false;
}
body->numSides = src.ParseInt();
if ( !src.ExpectTokenString( ")" ) ) {
return false;
}
} else if ( !token.Icmp( "bone" ) ) {
body->modelType = TRM_BONE;
if ( !src.ExpectTokenString( "(" ) ||
!body->v1.Parse( src ) ||
!src.ExpectTokenString( "," ) ||
!body->v2.Parse( src ) ||
!src.ExpectTokenString( "," ) ) {
return false;
}
body->width = src.ParseFloat();
if ( !src.ExpectTokenString( ")" ) ) {
return false;
}
} else if ( !token.Icmp( "custom" ) ) {
src.Error( "custom models not yet implemented" );
return false;
} else {
src.Error( "unkown model type %s", token.c_str() );
return false;
}
} else if ( !token.Icmp( "origin" ) ) {
if ( !body->origin.Parse( src ) ) {
return false;
}
} else if ( !token.Icmp( "angles" ) ) {
if ( !angles.Parse( src ) ) {
return false;
}
body->angles = idAngles( angles.ToVec3().x, angles.ToVec3().y, angles.ToVec3().z );
} else if ( !token.Icmp( "joint" ) ) {
if ( !src.ExpectTokenType( TT_STRING, 0, &token ) ) {
return false;
}
body->jointName = token;
hasJoint = true;
} else if ( !token.Icmp( "mod" ) ) {
if ( !src.ExpectAnyToken( &token ) ) {
return false;
}
body->jointMod = JointModFromString( token.c_str() );
} else if ( !token.Icmp( "density" ) ) {
body->density = src.ParseFloat();
} else if ( !token.Icmp( "inertiaScale" ) ) {
src.Parse1DMatrix( 9, body->inertiaScale[0].ToFloatPtr() );
} else if ( !token.Icmp( "friction" ) ) {
body->linearFriction = src.ParseFloat();
src.ExpectTokenString( "," );
body->angularFriction = src.ParseFloat();
src.ExpectTokenString( "," );
body->contactFriction = src.ParseFloat();
} else if ( !token.Icmp( "contents" ) ) {
ParseContents( src, body->contents );
} else if ( !token.Icmp( "clipMask" ) ) {
ParseContents( src, body->clipMask );
body->clipMask &= ~CONTENTS_CORPSE; // never allow collisions against corpses
} else if ( !token.Icmp( "selfCollision" ) ) {
body->selfCollision = src.ParseBool();
} else if ( !token.Icmp( "containedjoints" ) ) {
if ( !src.ExpectTokenType( TT_STRING, 0, &token ) ) {
return false;
}
body->containedJoints = token;
} else if ( !token.Icmp( "frictionDirection" ) ) {
if ( !body->frictionDirection.Parse( src ) ) {
return false;
}
} else if ( !token.Icmp( "contactMotorDirection" ) ) {
if ( !body->contactMotorDirection.Parse( src ) ) {
return false;
}
} else if ( token == "}" ) {
break;
} else {
src.Error( "unknown token %s in body", token.c_str() );
return false;
}
}
if ( body->modelType == TRM_INVALID ) {
src.Error( "no model set for body" );
return false;
}
if ( !hasJoint ) {
src.Error( "no joint set for body" );
return false;
}
body->clipMask |= CONTENTS_MOVEABLECLIP;
return true;
}
/*
================
idDeclAF::ParseFixed
================
*/
bool idDeclAF::ParseFixed( idLexer &src ) {
idToken token;
idDeclAF_Constraint *constraint = new (TAG_DECL) idDeclAF_Constraint;
constraint->SetDefault( this );
constraints.Alloc() = constraint;
if ( !src.ExpectTokenType( TT_STRING, 0, &token ) ||
!src.ExpectTokenString( "{" ) ) {
return false;
}
constraint->type = DECLAF_CONSTRAINT_FIXED;
constraint->name = token;
while( src.ReadToken( &token ) ) {
if ( !token.Icmp( "body1" ) ) {
src.ExpectTokenType( TT_STRING, 0, &token );
constraint->body1 = token;
} else if ( !token.Icmp( "body2" ) ) {
src.ExpectTokenType( TT_STRING, 0, &token );
constraint->body2 = token;
} else if ( token == "}" ) {
break;
} else {
src.Error( "unknown token %s in ball and socket joint", token.c_str() );
return false;
}
}
return true;
}
/*
================
idDeclAF::ParseBallAndSocketJoint
================
*/
bool idDeclAF::ParseBallAndSocketJoint( idLexer &src ) {
idToken token;
idDeclAF_Constraint *constraint = new (TAG_DECL) idDeclAF_Constraint;
constraint->SetDefault( this );
constraints.Alloc() = constraint;
if ( !src.ExpectTokenType( TT_STRING, 0, &token ) ||
!src.ExpectTokenString( "{" ) ) {
return false;
}
constraint->type = DECLAF_CONSTRAINT_BALLANDSOCKETJOINT;
constraint->limit = idDeclAF_Constraint::LIMIT_NONE;
constraint->name = token;
constraint->friction = 0.5f;
constraint->anchor.ToVec3().Zero();
constraint->shaft[0].ToVec3().Zero();
while( src.ReadToken( &token ) ) {
if ( !token.Icmp( "body1" ) ) {
src.ExpectTokenType( TT_STRING, 0, &token );
constraint->body1 = token;
} else if ( !token.Icmp( "body2" ) ) {
src.ExpectTokenType( TT_STRING, 0, &token );
constraint->body2 = token;
} else if ( !token.Icmp( "anchor" ) ) {
if ( !constraint->anchor.Parse( src ) ) {
return false;
}
} else if ( !token.Icmp( "conelimit" ) ) {
if ( !constraint->limitAxis.Parse( src ) ||
!src.ExpectTokenString( "," ) ) {
return false;
}
constraint->limitAngles[0] = src.ParseFloat();
if ( !src.ExpectTokenString( "," ) ||
!constraint->shaft[0].Parse( src ) ) {
return false;
}
constraint->limit = idDeclAF_Constraint::LIMIT_CONE;
} else if ( !token.Icmp( "pyramidlimit" ) ) {
if ( !constraint->limitAxis.Parse( src ) ||
!src.ExpectTokenString( "," ) ) {
return false;
}
constraint->limitAngles[0] = src.ParseFloat();
if ( !src.ExpectTokenString( "," ) ) {
return false;
}
constraint->limitAngles[1] = src.ParseFloat();
if ( !src.ExpectTokenString( "," ) ) {
return false;
}
constraint->limitAngles[2] = src.ParseFloat();
if ( !src.ExpectTokenString( "," ) ||
!constraint->shaft[0].Parse( src ) ) {
return false;
}
constraint->limit = idDeclAF_Constraint::LIMIT_PYRAMID;
} else if ( !token.Icmp( "friction" ) ) {
constraint->friction = src.ParseFloat();
} else if ( token == "}" ) {
break;
} else {
src.Error( "unknown token %s in ball and socket joint", token.c_str() );
return false;
}
}
return true;
}
/*
================
idDeclAF::ParseUniversalJoint
================
*/
bool idDeclAF::ParseUniversalJoint( idLexer &src ) {
idToken token;
idDeclAF_Constraint *constraint = new (TAG_DECL) idDeclAF_Constraint;
constraint->SetDefault( this );
constraints.Alloc() = constraint;
if ( !src.ExpectTokenType( TT_STRING, 0, &token ) ||
!src.ExpectTokenString( "{" ) ) {
return false;
}
constraint->type = DECLAF_CONSTRAINT_UNIVERSALJOINT;
constraint->limit = idDeclAF_Constraint::LIMIT_NONE;
constraint->name = token;
constraint->friction = 0.5f;
constraint->anchor.ToVec3().Zero();
constraint->shaft[0].ToVec3().Zero();
constraint->shaft[1].ToVec3().Zero();
while( src.ReadToken( &token ) ) {
if ( !token.Icmp( "body1" ) ) {
src.ExpectTokenType( TT_STRING, 0, &token );
constraint->body1 = token;
} else if ( !token.Icmp( "body2" ) ) {
src.ExpectTokenType( TT_STRING, 0, &token );
constraint->body2 = token;
} else if ( !token.Icmp( "anchor" ) ) {
if ( !constraint->anchor.Parse( src ) ) {
return false;
}
} else if ( !token.Icmp( "shafts" ) ) {
if ( !constraint->shaft[0].Parse( src ) ||
!src.ExpectTokenString( "," ) ||
!constraint->shaft[1].Parse( src ) ) {
return false;
}
} else if ( !token.Icmp( "conelimit" ) ) {
if ( !constraint->limitAxis.Parse( src ) ||
!src.ExpectTokenString( "," ) ) {
return false;
}
constraint->limitAngles[0] = src.ParseFloat();
constraint->limit = idDeclAF_Constraint::LIMIT_CONE;
} else if ( !token.Icmp( "pyramidlimit" ) ) {
if ( !constraint->limitAxis.Parse( src ) ||
!src.ExpectTokenString( "," ) ) {
return false;
}
constraint->limitAngles[0] = src.ParseFloat();
if ( !src.ExpectTokenString( "," ) ) {
return false;
}
constraint->limitAngles[1] = src.ParseFloat();
if ( !src.ExpectTokenString( "," ) ) {
return false;
}
constraint->limitAngles[2] = src.ParseFloat();
constraint->limit = idDeclAF_Constraint::LIMIT_PYRAMID;
} else if ( !token.Icmp( "friction" ) ) {
constraint->friction = src.ParseFloat();
} else if ( token == "}" ) {
break;
} else {
src.Error( "unknown token %s in universal joint", token.c_str() );
return false;
}
}
return true;
}
/*
================
idDeclAF::ParseHinge
================
*/
bool idDeclAF::ParseHinge( idLexer &src ) {
idToken token;
idDeclAF_Constraint *constraint = new (TAG_DECL) idDeclAF_Constraint;
constraint->SetDefault( this );
constraints.Alloc() = constraint;
if ( !src.ExpectTokenType( TT_STRING, 0, &token ) ||
!src.ExpectTokenString( "{" ) ) {
return false;
}
constraint->type = DECLAF_CONSTRAINT_HINGE;
constraint->limit = idDeclAF_Constraint::LIMIT_NONE;
constraint->name = token;
constraint->friction = 0.5f;
constraint->anchor.ToVec3().Zero();
constraint->axis.ToVec3().Zero();
while( src.ReadToken( &token ) ) {
if ( !token.Icmp( "body1" ) ) {
src.ExpectTokenType( TT_STRING, 0, &token );
constraint->body1 = token;
} else if ( !token.Icmp( "body2" ) ) {
src.ExpectTokenType( TT_STRING, 0, &token );
constraint->body2 = token;
} else if ( !token.Icmp( "anchor" ) ) {
if ( !constraint->anchor.Parse( src ) ) {
return false;
}
} else if ( !token.Icmp( "axis" ) ) {
if ( !constraint->axis.Parse( src ) ) {
return false;
}
} else if ( !token.Icmp( "limit" ) ) {
constraint->limitAngles[0] = src.ParseFloat();
if ( !src.ExpectTokenString( "," ) ) {
return false;
}
constraint->limitAngles[1] = src.ParseFloat();
if ( !src.ExpectTokenString( "," ) ) {
return false;
}
constraint->limitAngles[2] = src.ParseFloat();
constraint->limit = idDeclAF_Constraint::LIMIT_CONE;
} else if ( !token.Icmp( "friction" ) ) {
constraint->friction = src.ParseFloat();
} else if ( token == "}" ) {
break;
} else {
src.Error( "unknown token %s in hinge", token.c_str() );
return false;
}
}
return true;
}
/*
================
idDeclAF::ParseSlider
================
*/
bool idDeclAF::ParseSlider( idLexer &src ) {
idToken token;
idDeclAF_Constraint *constraint = new (TAG_DECL) idDeclAF_Constraint;
constraint->SetDefault( this );
constraints.Alloc() = constraint;
if ( !src.ExpectTokenType( TT_STRING, 0, &token ) ||
!src.ExpectTokenString( "{" ) ) {
return false;
}
constraint->type = DECLAF_CONSTRAINT_SLIDER;
constraint->limit = idDeclAF_Constraint::LIMIT_NONE;
constraint->name = token;
constraint->friction = 0.5f;
while( src.ReadToken( &token ) ) {
if ( !token.Icmp( "body1" ) ) {
src.ExpectTokenType( TT_STRING, 0, &token );
constraint->body1 = token;
} else if ( !token.Icmp( "body2" ) ) {
src.ExpectTokenType( TT_STRING, 0, &token );
constraint->body2 = token;
} else if ( !token.Icmp( "axis" ) ) {
if ( !constraint->axis.Parse( src ) ) {
return false;
}
} else if ( !token.Icmp( "friction" ) ) {
constraint->friction = src.ParseFloat();
} else if ( token == "}" ) {
break;
} else {
src.Error( "unknown token %s in slider", token.c_str() );
return false;
}
}
return true;
}
/*
================
idDeclAF::ParseSpring
================
*/
bool idDeclAF::ParseSpring( idLexer &src ) {
idToken token;
idDeclAF_Constraint *constraint = new (TAG_DECL) idDeclAF_Constraint;
constraint->SetDefault( this );
constraints.Alloc() = constraint;
if ( !src.ExpectTokenType( TT_STRING, 0, &token ) ||
!src.ExpectTokenString( "{" ) ) {
return false;
}
constraint->type = DECLAF_CONSTRAINT_SPRING;
constraint->limit = idDeclAF_Constraint::LIMIT_NONE;
constraint->name = token;
constraint->friction = 0.5f;
while( src.ReadToken( &token ) ) {
if ( !token.Icmp( "body1" ) ) {
src.ExpectTokenType( TT_STRING, 0, &token );
constraint->body1 = token;
} else if ( !token.Icmp( "body2" ) ) {
src.ExpectTokenType( TT_STRING, 0, &token );
constraint->body2 = token;
} else if ( !token.Icmp( "anchor1" ) ) {
if ( !constraint->anchor.Parse( src ) ) {
return false;
}
} else if ( !token.Icmp( "anchor2" ) ) {
if ( !constraint->anchor2.Parse( src ) ) {
return false;
}
} else if ( !token.Icmp( "friction" ) ) {
constraint->friction = src.ParseFloat();
} else if ( !token.Icmp( "stretch" ) ) {
constraint->stretch = src.ParseFloat();
} else if ( !token.Icmp( "compress" ) ) {
constraint->compress = src.ParseFloat();
} else if ( !token.Icmp( "damping" ) ) {
constraint->damping = src.ParseFloat();
} else if ( !token.Icmp( "restLength" ) ) {
constraint->restLength = src.ParseFloat();
} else if ( !token.Icmp( "minLength" ) ) {
constraint->minLength = src.ParseFloat();
} else if ( !token.Icmp( "maxLength" ) ) {
constraint->maxLength = src.ParseFloat();
} else if ( token == "}" ) {
break;
} else {
src.Error( "unknown token %s in spring", token.c_str() );
return false;
}
}
return true;
}
/*
================
idDeclAF::ParseSettings
================
*/
bool idDeclAF::ParseSettings( idLexer &src ) {
idToken token;
if ( !src.ExpectTokenString( "{" ) ) {
return false;
}
while( src.ReadToken( &token ) ) {
if ( !token.Icmp( "mesh" ) ) {
if ( !src.ExpectTokenType( TT_STRING, 0, &token ) ) {
return false;
}
} else if ( !token.Icmp( "anim" ) ) {
if ( !src.ExpectTokenType( TT_STRING, 0, &token ) ) {
return false;
}
} else if ( !token.Icmp( "model" ) ) {
if ( !src.ExpectTokenType( TT_STRING, 0, &token ) ) {
return false;
}
model = token;
} else if ( !token.Icmp( "skin" ) ) {
if ( !src.ExpectTokenType( TT_STRING, 0, &token ) ) {
return false;
}
skin = token;
} else if ( !token.Icmp( "friction" ) ) {
defaultLinearFriction = src.ParseFloat();
if ( !src.ExpectTokenString( "," ) ) {
return false;
}
defaultAngularFriction = src.ParseFloat();
if ( !src.ExpectTokenString( "," ) ) {
return false;
}
defaultContactFriction = src.ParseFloat();
if ( src.CheckTokenString( "," ) ) {
defaultConstraintFriction = src.ParseFloat();
}
} else if ( !token.Icmp( "totalMass" ) ) {
totalMass = src.ParseFloat();
} else if ( !token.Icmp( "suspendSpeed" ) ) {
suspendVelocity[0] = src.ParseFloat();
if ( !src.ExpectTokenString( "," ) ) {
return false;
}
suspendVelocity[1] = src.ParseFloat();
if ( !src.ExpectTokenString( "," ) ) {
return false;
}
suspendAcceleration[0] = src.ParseFloat();
if ( !src.ExpectTokenString( "," ) ) {
return false;
}
suspendAcceleration[1] = src.ParseFloat();
} else if ( !token.Icmp( "noMoveTime" ) ) {
noMoveTime = src.ParseFloat();
} else if ( !token.Icmp( "noMoveTranslation" ) ) {
noMoveTranslation = src.ParseFloat();
} else if ( !token.Icmp( "noMoveRotation" ) ) {
noMoveRotation = src.ParseFloat();
} else if ( !token.Icmp( "minMoveTime" ) ) {
minMoveTime = src.ParseFloat();
} else if ( !token.Icmp( "maxMoveTime" ) ) {
maxMoveTime = src.ParseFloat();
} else if ( !token.Icmp( "contents" ) ) {
ParseContents( src, contents );
} else if ( !token.Icmp( "clipMask" ) ) {
ParseContents( src, clipMask );
clipMask &= ~CONTENTS_CORPSE; // never allow collisions against corpses
} else if ( !token.Icmp( "selfCollision" ) ) {
selfCollision = src.ParseBool();
} else if ( token == "}" ) {
break;
} else {
src.Error( "unknown token %s in settings", token.c_str() );
return false;
}
}
return true;
}
/*
================
idDeclAF::Parse
================
*/
bool idDeclAF::Parse( const char *text, const int textLength, bool allowBinaryVersion ) {
int i, j;
idLexer src;
idToken token;
src.LoadMemory( text, textLength, GetFileName(), GetLineNum() );
src.SetFlags( DECL_LEXER_FLAGS );
src.SkipUntilString( "{" );
while( src.ReadToken( &token ) ) {
if ( !token.Icmp( "settings" ) ) {
if ( !ParseSettings( src ) ) {
return false;
}
} else if ( !token.Icmp( "body" ) ) {
if ( !ParseBody( src ) ) {
return false;
}
} else if ( !token.Icmp( "fixed" ) ) {
if ( !ParseFixed( src ) ) {
return false;
}
} else if ( !token.Icmp( "ballAndSocketJoint" ) ) {
if ( !ParseBallAndSocketJoint( src ) ) {
return false;
}
} else if ( !token.Icmp( "universalJoint" ) ) {
if ( !ParseUniversalJoint( src ) ) {
return false;
}
} else if ( !token.Icmp( "hinge" ) ) {
if ( !ParseHinge( src ) ) {
return false;
}
} else if ( !token.Icmp( "slider" ) ) {
if ( !ParseSlider( src ) ) {
return false;
}
} else if ( !token.Icmp( "spring" ) ) {
if ( !ParseSpring( src ) ) {
return false;
}
} else if ( token == "}" ) {
break;
} else {
src.Error( "unknown keyword %s", token.c_str() );
return false;
}
}
for ( i = 0; i < bodies.Num(); i++ ) {
// check for multiple bodies with the same name
for ( j = i+1; j < bodies.Num(); j++ ) {
if ( bodies[i]->name == bodies[j]->name ) {
src.Error( "two bodies with the same name \"%s\"", bodies[i]->name.c_str() );
}
}
}
for ( i = 0; i < constraints.Num(); i++ ) {
// check for multiple constraints with the same name
for ( j = i+1; j < constraints.Num(); j++ ) {
if ( constraints[i]->name == constraints[j]->name ) {
src.Error( "two constraints with the same name \"%s\"", constraints[i]->name.c_str() );
}
}
// check if there are two valid bodies set
if ( constraints[i]->body1 == "" ) {
src.Error( "no valid body1 specified for constraint '%s'", constraints[i]->name.c_str() );
}
if ( constraints[i]->body2 == "" ) {
src.Error( "no valid body2 specified for constraint '%s'", constraints[i]->name.c_str() );
}
}
// make sure the body which modifies the origin comes first
for ( i = 0; i < bodies.Num(); i++ ) {
if ( bodies[i]->jointName == "origin" ) {
if ( i != 0 ) {
idDeclAF_Body *b = bodies[0];
bodies[0] = bodies[i];
bodies[i] = b;
}
break;
}
}
return true;
}
/*
================
idDeclAF::DefaultDefinition
================
*/
const char *idDeclAF::DefaultDefinition() const {
return
"{\n"
"\t" "settings {\n"
"\t\t" "model \"\"\n"
"\t\t" "skin \"\"\n"
"\t\t" "friction 0.01, 0.01, 0.8, 0.5\n"
"\t\t" "suspendSpeed 20, 30, 40, 60\n"
"\t\t" "noMoveTime 1\n"
"\t\t" "noMoveTranslation 10\n"
"\t\t" "noMoveRotation 10\n"
"\t\t" "minMoveTime -1\n"
"\t\t" "maxMoveTime -1\n"
"\t\t" "totalMass -1\n"
"\t\t" "contents corpse\n"
"\t\t" "clipMask solid, corpse\n"
"\t\t" "selfCollision 1\n"
"\t" "}\n"
"\t" "body \"body\" {\n"
"\t\t" "joint \"origin\"\n"
"\t\t" "mod orientation\n"
"\t\t" "model box( ( -10, -10, -10 ), ( 10, 10, 10 ) )\n"
"\t\t" "origin ( 0, 0, 0 )\n"
"\t\t" "density 0.2\n"
"\t\t" "friction 0.01, 0.01, 0.8\n"
"\t\t" "contents corpse\n"
"\t\t" "clipMask solid, corpse\n"
"\t\t" "selfCollision 1\n"
"\t\t" "containedJoints \"*origin\"\n"
"\t" "}\n"
"}\n";
}
/*
================
idDeclAF::FreeData
================
*/
void idDeclAF::FreeData() {
modified = false;
defaultLinearFriction = 0.01f;
defaultAngularFriction = 0.01f;
defaultContactFriction = 0.8f;
defaultConstraintFriction = 0.5f;
totalMass = -1;
suspendVelocity.Set( 20.0f, 30.0f );
suspendAcceleration.Set( 40.0f, 60.0f );
noMoveTime = 1.0f;
noMoveTranslation = 10.0f;
noMoveRotation = 10.0f;
minMoveTime = -1.0f;
maxMoveTime = -1.0f;
selfCollision = true;
contents = CONTENTS_CORPSE;
clipMask = CONTENTS_SOLID;
bodies.DeleteContents( true );
constraints.DeleteContents( true );
}
/*
================
idDeclAF::Finish
================
*/
void idDeclAF::Finish( const getJointTransform_t GetJointTransform, const idJointMat *frame, void *model ) const {
int i;
const char *name = GetName();
for ( i = 0; i < bodies.Num(); i++ ) {
idDeclAF_Body *body = bodies[i];
body->v1.Finish( name, GetJointTransform, frame, model );
body->v2.Finish( name, GetJointTransform, frame, model );
body->origin.Finish( name, GetJointTransform, frame, model );
body->frictionDirection.Finish( name, GetJointTransform, frame, model );
body->contactMotorDirection.Finish( name, GetJointTransform, frame, model );
}
for ( i = 0; i < constraints.Num(); i++ ) {
idDeclAF_Constraint *constraint = constraints[i];
constraint->anchor.Finish( name, GetJointTransform, frame, model );
constraint->anchor2.Finish( name, GetJointTransform, frame, model );
constraint->shaft[0].Finish( name, GetJointTransform, frame, model );
constraint->shaft[1].Finish( name, GetJointTransform, frame, model );
constraint->axis.Finish( name, GetJointTransform, frame, model );
constraint->limitAxis.Finish( name, GetJointTransform, frame, model );
}
}
/*
================
idDeclAF::NewBody
================
*/
void idDeclAF::NewBody( const char *name ) {
idDeclAF_Body *body;
body = new (TAG_DECL) idDeclAF_Body();
body->SetDefault( this );
body->name = name;
bodies.Append( body );
}
/*
================
idDeclAF::RenameBody
rename the body with the given name and rename
all constraint body references
================
*/
void idDeclAF::RenameBody( const char *oldName, const char *newName ) {
int i;
for ( i = 0; i < bodies.Num(); i++ ) {
if ( bodies[i]->name.Icmp( oldName ) == 0 ) {
bodies[i]->name = newName;
break;
}
}
for ( i = 0; i < constraints.Num(); i++ ) {
if ( constraints[i]->body1.Icmp( oldName ) == 0 ) {
constraints[i]->body1 = newName;
} else if ( constraints[i]->body2.Icmp( oldName ) == 0 ) {
constraints[i]->body2 = newName;
}
}
}
/*
================
idDeclAF::DeleteBody
delete the body with the given name and delete
all constraints that reference the body
================
*/
void idDeclAF::DeleteBody( const char *name ) {
int i;
for ( i = 0; i < bodies.Num(); i++ ) {
if ( bodies[i]->name.Icmp( name ) == 0 ) {
delete bodies[i];
bodies.RemoveIndex( i );
break;
}
}
for ( i = 0; i < constraints.Num(); i++ ) {
if ( constraints[i]->body1.Icmp( name ) == 0 ||
constraints[i]->body2.Icmp( name ) == 0 ) {
delete constraints[i];
constraints.RemoveIndex( i );
i--;
}
}
}
/*
================
idDeclAF::NewConstraint
================
*/
void idDeclAF::NewConstraint( const char *name ) {
idDeclAF_Constraint *constraint;
constraint = new (TAG_DECL) idDeclAF_Constraint;
constraint->SetDefault( this );
constraint->name = name;
constraints.Append( constraint );
}
/*
================
idDeclAF::RenameConstraint
================
*/
void idDeclAF::RenameConstraint( const char *oldName, const char *newName ) {
int i;
for ( i = 0; i < constraints.Num(); i++ ) {
if ( constraints[i]->name.Icmp( oldName ) == 0 ) {
constraints[i]->name = newName;
return;
}
}
}
/*
================
idDeclAF::DeleteConstraint
================
*/
void idDeclAF::DeleteConstraint( const char *name ) {
int i;
for ( i = 0; i < constraints.Num(); i++ ) {
if ( constraints[i]->name.Icmp( name ) == 0 ) {
delete constraints[i];
constraints.RemoveIndex( i );
return;
}
}
}
/*
================
idDeclAF::idDeclAF
================
*/
idDeclAF::idDeclAF() {
FreeData();
}
/*
================
idDeclAF::~idDeclAF
================
*/
idDeclAF::~idDeclAF() {
bodies.DeleteContents( true );
constraints.DeleteContents( true );
}