46#include "models/FGGroundReactions.h"
47#include "math/FGTable.h"
48#include "input_output/FGXMLElement.h"
49#include "models/FGInertial.h"
65const FGMatrix33 FGLGear::Tb2s(-1./inchtoft, 0., 0., 0., 1./inchtoft, 0., 0., 0., -1./inchtoft);
66const FGMatrix33 FGLGear::Ts2b(-inchtoft, 0., 0., 0., inchtoft, 0., 0., 0., -inchtoft);
78 StaticFriction(false),
81 kSpring = bDamp = bDampRebound = dynamicFCoeff = staticFCoeff = rollingFCoeff = maxSteerAngle = 0;
82 isRetractable =
false;
84 eDampTypeRebound = dtLinear;
88 if (sContactType ==
"BOGEY") {
89 eContactType = ctBOGEY;
90 }
else if (sContactType ==
"STRUCTURE") {
91 eContactType = ctSTRUCTURE;
94 eContactType = ctSTRUCTURE;
98 if (eContactType == ctSTRUCTURE) {
99 kSpring = in.EmptyWeight;
101 bDampRebound = kSpring * 10;
112 fStrutForce =
new FGFunction(fdmex, springFunc);
120 eDampType = dtSquare;
130 eDampTypeRebound = dtSquare;
136 bDampRebound = bDamp;
137 eDampTypeRebound = eDampType;
155 if ((maxSteerAngle == 360 && !castered_el)
157 eSteerType = stCaster;
160 else if (maxSteerAngle == 0.0) {
161 eSteerType = stFixed;
164 eSteerType = stSteer;
170 while (force_table) {
172 if (force_type ==
"CORNERING_COEFF") {
173 ForceY_Table =
new FGTable(PropertyManager, force_table);
176 cerr <<
"Undefined force table for " << name <<
" contact point" << endl;
185 s <<
"No location given for contact " << name;
186 cerr << endl << s.str() << endl;
189 SetTransformType(FGForce::tCustom);
192 if (element && (eContactType == ctBOGEY)) {
195 mTGear = quatFromEuler.
GetT();
205 if (sBrakeGroup ==
"LEFT" ) eBrakeGrp = bgLeft;
206 else if (sBrakeGroup ==
"RIGHT" ) eBrakeGrp = bgRight;
207 else if (sBrakeGroup ==
"CENTER") eBrakeGrp = bgCenter;
208 else if (sBrakeGroup ==
"NOSE" ) eBrakeGrp = bgCenter;
209 else if (sBrakeGroup ==
"TAIL" ) eBrakeGrp = bgCenter;
210 else if (sBrakeGroup ==
"NONE" ) eBrakeGrp = bgNone;
211 else if (sBrakeGroup.empty() ) eBrakeGrp = bgNone;
213 cerr <<
"Improper braking group specification in config file: "
214 << sBrakeGroup <<
" is undefined." << endl;
220 useFCSGearPos =
false;
222 TakeoffReported = LandingReported =
false;
233 bind(PropertyManager.get());
249void FGLGear::ResetToIC(
void)
253 WOW = lastWOW =
false;
254 FirstContact =
false;
255 StartedGroundRun =
false;
256 LandingDistanceTraveled = TakeoffDistanceTraveled = TakeoffDistanceTraveled50ft = 0.0;
257 MaximumStrutForce = MaximumStrutTravel = 0.0;
258 SinkRate = GroundSpeed = 0.0;
261 vWhlVelVec.InitMatrix();
263 compressLength = 0.0;
270 for (
int i=0; i < 3; i++) {
271 LMultiplier[i].ForceJacobian.InitMatrix();
272 LMultiplier[i].LeverArm.InitMatrix();
273 LMultiplier[i].Min = 0.0;
274 LMultiplier[i].Max = 0.0;
275 LMultiplier[i].value = 0.0;
283 double gearPos = 1.0;
292 vLocalGear = in.Tb2l * vWhlBodyVec;
297 double height = fdmex->
GetInertial()->GetContactPoint(gearLoc, contact,
298 normal, terrainVel, dummy);
302 AGL = max(height, 0.0);
304 if (isRetractable) gearPos = GetGearUnitPos();
306 if (gearPos > 0.99) {
308 if (!fdmex->GetTrimStatus())
314 isSolid = GroundReactions->
GetSolid();
317 double LGearProj = 1.0;
321 vGroundNormal = in.Tec2b * normal;
327 double normalZ = (in.Tec2l*normal)(eZ);
328 LGearProj = -(mTGear.
Transposed() * vGroundNormal)(eZ);
332 switch(eContactType) {
335 compressLength = LGearProj > 0.0 ? height * normalZ / LGearProj : 0.0;
343 compressLength = height * normalZ / DotProduct(normal, normal);
344 vWhlDisplVec = compressLength * vGroundNormal;
353 vActingXYZn = vXYZn + Tb2s * vWhlDisplVec;
355 vBodyWhlVel += in.UVW - in.Tec2b * terrainVel;
356 vWhlVelVec = mTGear.
Transposed() * vBodyWhlVel;
358 InitializeReporting();
359 ComputeSteeringAngle();
360 ComputeGroundFrame();
362 vGroundWhlVel = mT.
Transposed() * vBodyWhlVel;
364 if (fdmex->GetTrimStatus() || in.TotalDeltaT == 0.0)
367 compressSpeed = -vGroundWhlVel(eZ);
368 if (eContactType == ctBOGEY)
369 compressSpeed /= LGearProj;
374 double maxCompressSpeed = compressLength/in.TotalDeltaT;
375 if (fabs(compressSpeed) > maxCompressSpeed)
376 compressSpeed = sign(compressSpeed)*maxCompressSpeed;
379 ComputeVerticalStrutForce();
382 if (eContactType == ctBOGEY) {
384 ComputeBrakeForceCoefficient();
385 ComputeSideForceCoefficient();
390 ComputeJacobian(vWhlContactVec);
392 compressLength = 0.0;
396 vWhlDisplVec.InitMatrix();
398 LMultiplier[ftRoll].value = 0.0;
399 LMultiplier[ftSide].value = 0.0;
400 LMultiplier[ftDynamic].value = 0.0;
403 SteerAngle *= max(gearPos-0.8, 0.0)/0.2;
411 vWhlVelVec(eX) -= 13.0 * in.TotalDeltaT;
412 if (vWhlVelVec(eX) < 0.0) vWhlVelVec(eX) = 0.0;
415 if (!fdmex->GetTrimStatus()) {
416 ReportTakeoffOrLanding();
420 if (WOW && lastWOW) CrashDetect();
425 return FGForce::GetBodyForces();
434void FGLGear::ComputeGroundFrame(
void)
439 roll -= DotProduct(roll, vGroundNormal) * vGroundNormal;
443 mT(eX,eX) = roll(eX);
444 mT(eY,eX) = roll(eY);
445 mT(eZ,eX) = roll(eZ);
446 mT(eX,eY) = side(eX);
447 mT(eY,eY) = side(eY);
448 mT(eZ,eY) = side(eZ);
449 mT(eX,eZ) = vGroundNormal(eX);
450 mT(eY,eZ) = vGroundNormal(eY);
451 mT(eZ,eZ) = vGroundNormal(eZ);
457void FGLGear::ComputeSlipAngle(
void)
460 if (vGroundWhlVel.
Magnitude(eX,eY) > 1E-3)
461 WheelSlip = -atan2(vGroundWhlVel(eY), fabs(vGroundWhlVel(eX)))*radtodeg;
468void FGLGear::ComputeSteeringAngle(
void)
473 SteerAngle = atan2(vWhlVelVec(eY), fabs(vWhlVelVec(eX)));
480void FGLGear::ResetReporting(
void)
482 if (in.DistanceAGL > 200.0) {
483 FirstContact =
false;
484 StartedGroundRun =
false;
485 LandingReported =
false;
486 TakeoffReported =
true;
487 LandingDistanceTraveled = 0.0;
488 MaximumStrutForce = MaximumStrutTravel = 0.0;
494void FGLGear::InitializeReporting(
void)
501 SinkRate = compressSpeed;
502 GroundSpeed = in.Vground;
503 TakeoffReported =
false;
508 if ((in.Vground > 0.1) &&
509 (in.BrakePos[bgLeft] == 0) &&
510 (in.BrakePos[bgRight] == 0) &&
511 (in.TakeoffThrottle && !StartedGroundRun))
513 TakeoffDistanceTraveled = 0;
514 TakeoffDistanceTraveled50ft = 0;
515 StartedGroundRun =
true;
522void FGLGear::ReportTakeoffOrLanding(
void)
525 LandingDistanceTraveled += in.Vground * in.TotalDeltaT;
527 if (StartedGroundRun) {
528 TakeoffDistanceTraveled50ft += in.Vground * in.TotalDeltaT;
529 if (WOW) TakeoffDistanceTraveled += in.Vground * in.TotalDeltaT;
533 && in.Vground <= 0.05
537 if (debug_lvl > 0) Report(erLand);
542 && (in.DistanceAGL - vLocalGear(eZ)) > 50.0
545 if (debug_lvl > 0) Report(erTakeoff);
551 cout <<
"GEAR_CONTACT: " << fdmex->
GetSimTime() <<
" seconds: " << name
552 <<
" " << WOW << endl;
560void FGLGear::CrashDetect(
void)
562 if ( (compressLength > 500.0 ||
564 GetMoments().Magnitude() > 5000000000.0 ||
568 cout <<
"*CRASH DETECTED* " << fdmex->
GetSimTime() <<
" seconds: " << name;
583void FGLGear::ComputeBrakeForceCoefficient(
void)
585 BrakeFCoeff = rollingFFactor * rollingFCoeff;
587 if (eBrakeGrp != bgNone)
588 BrakeFCoeff += in.BrakePos[eBrakeGrp] * staticFFactor * (staticFCoeff - rollingFCoeff);
601void FGLGear::ComputeSideForceCoefficient(
void)
604 FCoeff = ForceY_Table->
GetValue(WheelSlip);
606 double StiffSlip = Stiffness*WheelSlip;
607 FCoeff = Peak * sin(Shape*atan(StiffSlip - Curvature*(StiffSlip - atan(StiffSlip))));
609 FCoeff *= staticFFactor;
619void FGLGear::ComputeVerticalStrutForce()
622 StrutForce = min(fStrutForce->
GetValue(), (
double)0.0);
624 double springForce = -compressLength * kSpring;
625 double dampForce = 0;
627 if (compressSpeed >= 0.0) {
629 if (eDampType == dtLinear)
630 dampForce = -compressSpeed * bDamp;
632 dampForce = -compressSpeed * compressSpeed * bDamp;
636 if (eDampTypeRebound == dtLinear)
637 dampForce = -compressSpeed * bDampRebound;
639 dampForce = compressSpeed * compressSpeed * bDampRebound;
643 StrutForce = min(springForce + dampForce, (
double)0.0);
644 if (StrutForce > maximumForce) {
645 StrutForce = maximumForce;
646 compressLength = -StrutForce / kSpring;
651 switch (eContactType) {
654 vFn(eZ) = StrutForce / (mTGear.
Transposed()*vGroundNormal)(eZ);
657 vFn(eZ) = -StrutForce;
662 MaximumStrutForce = max(MaximumStrutForce, fabs(StrutForce));
663 MaximumStrutTravel = max(MaximumStrutTravel, fabs(compressLength));
668double FGLGear::GetGearUnitPos(
void)
const
671 if( useFCSGearPos || in.FCSGearPos != 1.0 ) {
672 useFCSGearPos =
true;
673 return in.FCSGearPos;
682void FGLGear::ComputeJacobian(
const FGColumnVector3& vWhlContactVec)
687 if ((eContactType == ctSTRUCTURE) && (vGroundWhlVel.
Magnitude(eX,eY) > 1E-3)) {
689 FGColumnVector3 velocityDirection = vGroundWhlVel;
691 StaticFriction =
false;
693 velocityDirection(eZ) = 0.;
694 velocityDirection.Normalize();
696 LMultiplier[ftDynamic].ForceJacobian = mT * velocityDirection;
697 LMultiplier[ftDynamic].Max = 0.;
698 LMultiplier[ftDynamic].Min = -fabs(staticFFactor * dynamicFCoeff * vFn(eZ));
699 LMultiplier[ftDynamic].LeverArm = vWhlContactVec;
706 LMultiplier[ftDynamic].value =
Constrain(LMultiplier[ftDynamic].Min, LMultiplier[ftDynamic].value, LMultiplier[ftDynamic].Max);
708 GroundReactions->RegisterLagrangeMultiplier(&LMultiplier[ftDynamic]);
716 StaticFriction =
true;
718 LMultiplier[ftRoll].ForceJacobian = mT * FGColumnVector3(1.,0.,0.);
719 LMultiplier[ftSide].ForceJacobian = mT * FGColumnVector3(0.,1.,0.);
720 LMultiplier[ftRoll].LeverArm = vWhlContactVec;
721 LMultiplier[ftSide].LeverArm = vWhlContactVec;
723 switch(eContactType) {
725 LMultiplier[ftRoll].Max = fabs(BrakeFCoeff * vFn(eZ));
726 LMultiplier[ftSide].Max = fabs(FCoeff * vFn(eZ));
729 LMultiplier[ftRoll].Max = fabs(staticFFactor * staticFCoeff * vFn(eZ));
730 LMultiplier[ftSide].Max = LMultiplier[ftRoll].Max;
734 LMultiplier[ftRoll].Min = -LMultiplier[ftRoll].Max;
735 LMultiplier[ftSide].Min = -LMultiplier[ftSide].Max;
742 LMultiplier[ftRoll].value =
Constrain(LMultiplier[ftRoll].Min, LMultiplier[ftRoll].value, LMultiplier[ftRoll].Max);
743 LMultiplier[ftSide].value =
Constrain(LMultiplier[ftSide].Min, LMultiplier[ftSide].value, LMultiplier[ftSide].Max);
745 GroundReactions->RegisterLagrangeMultiplier(&LMultiplier[ftRoll]);
746 GroundReactions->RegisterLagrangeMultiplier(&LMultiplier[ftSide]);
754void FGLGear::UpdateForces(
void)
756 if (StaticFriction) {
757 vFn(eX) = LMultiplier[ftRoll].value;
758 vFn(eY) = LMultiplier[ftSide].value;
761 FGColumnVector3 forceDir = mT.
Transposed() * LMultiplier[ftDynamic].ForceJacobian;
762 vFn(eX) = LMultiplier[ftDynamic].value * forceDir(eX);
763 vFn(eY) = LMultiplier[ftDynamic].value * forceDir(eY);
769void FGLGear::SetstaticFCoeff(
double coeff)
771 staticFCoeff = coeff;
777void FGLGear::bind(FGPropertyManager* PropertyManager)
779 string property_name;
780 string base_property_name;
782 switch(eContactType) {
784 base_property_name = CreateIndexedPropertyName(
"gear/unit", GearNumber);
787 base_property_name = CreateIndexedPropertyName(
"contact/unit", GearNumber);
793 property_name = base_property_name +
"/AGL-ft";
794 PropertyManager->Tie(property_name.c_str(), &AGL);
795 property_name = base_property_name +
"/WOW";
796 PropertyManager->Tie( property_name.c_str(), &WOW );
797 property_name = base_property_name +
"/x-position";
798 PropertyManager->Tie( property_name.c_str(), (FGForce*)
this,
799 &FGForce::GetLocationX, &FGForce::SetLocationX);
800 property_name = base_property_name +
"/y-position";
801 PropertyManager->Tie( property_name.c_str(), (FGForce*)
this,
802 &FGForce::GetLocationY, &FGForce::SetLocationY);
803 property_name = base_property_name +
"/z-position";
804 PropertyManager->Tie( property_name.c_str(), (FGForce*)
this,
805 &FGForce::GetLocationZ, &FGForce::SetLocationZ);
806 property_name = base_property_name +
"/compression-ft";
807 PropertyManager->Tie( property_name.c_str(), &compressLength );
808 property_name = base_property_name +
"/compression-velocity-fps";
809 PropertyManager->Tie( property_name.c_str(), &compressSpeed );
810 property_name = base_property_name +
"/static_friction_coeff";
811 PropertyManager->Tie( property_name.c_str(), (FGLGear*)
this,
812 &FGLGear::GetstaticFCoeff, &FGLGear::SetstaticFCoeff);
813 property_name = base_property_name +
"/dynamic_friction_coeff";
814 PropertyManager->Tie( property_name.c_str(), &dynamicFCoeff );
816 if (eContactType == ctBOGEY) {
817 property_name = base_property_name +
"/slip-angle-deg";
818 PropertyManager->Tie( property_name.c_str(), &WheelSlip );
819 property_name = base_property_name +
"/wheel-speed-fps";
820 PropertyManager->Tie( property_name.c_str(), (FGLGear*)
this,
821 &FGLGear::GetWheelRollVel);
822 property_name = base_property_name +
"/side_friction_coeff";
823 PropertyManager->Tie( property_name.c_str(), &FCoeff );
824 property_name = base_property_name +
"/rolling_friction_coeff";
825 PropertyManager->Tie( property_name.c_str(), &rollingFCoeff );
827 if (eSteerType == stCaster) {
828 property_name = base_property_name +
"/steering-angle-deg";
829 PropertyManager->Tie( property_name.c_str(),
this, &FGLGear::GetSteerAngleDeg );
830 property_name = base_property_name +
"/castered";
831 PropertyManager->Tie( property_name.c_str(), &Castered);
835 if( isRetractable ) {
836 property_name = base_property_name +
"/pos-norm";
837 PropertyManager->Tie( property_name.c_str(), &GearPos );
840 if (eSteerType != stFixed) {
844 string tmp = CreateIndexedPropertyName(
"fcs/steer-pos-deg", GearNumber);
845 PropertyManager->Tie(tmp.c_str(),
this, &FGLGear::GetSteerAngleDeg, &FGLGear::SetSteerAngleDeg);
848 property_name = base_property_name +
"/solid";
849 PropertyManager->Tie( property_name.c_str(), &isSolid);
850 property_name = base_property_name +
"/bumpiness";
851 PropertyManager->Tie( property_name.c_str(), &bumpiness);
852 property_name = base_property_name +
"/maximum-force-lbs";
853 PropertyManager->Tie( property_name.c_str(), &maximumForce);
854 property_name = base_property_name +
"/rolling_friction-factor";
855 PropertyManager->Tie( property_name.c_str(), &rollingFFactor);
856 property_name = base_property_name +
"/static-friction-factor";
857 PropertyManager->Tie( property_name.c_str(), &staticFFactor);
862void FGLGear::Report(ReportType repType)
864 if (fabs(TakeoffDistanceTraveled) < 0.001)
return;
868 cout << endl <<
"Touchdown report for " << name <<
" (WOW at time: "
869 << fdmex->
GetSimTime() <<
" seconds)" << endl;
870 cout <<
" Sink rate at contact: " << SinkRate <<
" fps, "
871 << SinkRate*0.3048 <<
" mps" << endl;
872 cout <<
" Contact ground speed: " << GroundSpeed*.5925 <<
" knots, "
873 << GroundSpeed*0.3048 <<
" mps" << endl;
874 cout <<
" Maximum contact force: " << MaximumStrutForce <<
" lbs, "
875 << MaximumStrutForce*4.448 <<
" Newtons" << endl;
876 cout <<
" Maximum strut travel: " << MaximumStrutTravel*12.0 <<
" inches, "
877 << MaximumStrutTravel*30.48 <<
" cm" << endl;
878 cout <<
" Distance traveled: " << LandingDistanceTraveled <<
" ft, "
879 << LandingDistanceTraveled*0.3048 <<
" meters" << endl;
880 LandingReported =
true;
883 cout << endl <<
"Takeoff report for " << name <<
" (Liftoff at time: "
884 << fdmex->
GetSimTime() <<
" seconds)" << endl;
885 cout <<
" Distance traveled: " << TakeoffDistanceTraveled
886 <<
" ft, " << TakeoffDistanceTraveled*0.3048 <<
" meters" << endl;
887 cout <<
" Distance traveled (over 50'): " << TakeoffDistanceTraveled50ft
888 <<
" ft, " << TakeoffDistanceTraveled50ft*0.3048 <<
" meters" << endl;
889 cout <<
" [Altitude (ASL): " << in.DistanceASL <<
" ft. / "
890 << in.DistanceASL*FGJSBBase::fttom <<
" m | Temperature: "
891 << in.Temperature - 459.67 <<
" F / "
893 cout <<
" [Velocity (KCAS): " << in.VcalibratedKts <<
"]" << endl;
894 TakeoffReported =
true;
920void FGLGear::Debug(
int from)
922 static const char* sSteerType[] = {
"STEERABLE",
"FIXED",
"CASTERED" };
923 static const char* sBrakeGroup[] = {
"NONE",
"LEFT",
"RIGHT",
"CENTER",
"NOSE",
"TAIL"};
924 static const char* sContactType[] = {
"BOGEY",
"STRUCTURE" };
926 if (debug_lvl <= 0)
return;
930 cout <<
" " << sContactType[eContactType] <<
" " << name << endl;
931 cout <<
" Location: " << vXYZn << endl;
932 cout <<
" Spring Constant: " << kSpring << endl;
934 if (eDampType == dtLinear)
935 cout <<
" Damping Constant: " << bDamp <<
" (linear)" << endl;
937 cout <<
" Damping Constant: " << bDamp <<
" (square law)" << endl;
939 if (eDampTypeRebound == dtLinear)
940 cout <<
" Rebound Damping Constant: " << bDampRebound <<
" (linear)" << endl;
942 cout <<
" Rebound Damping Constant: " << bDampRebound <<
" (square law)" << endl;
944 cout <<
" Dynamic Friction: " << dynamicFCoeff << endl;
945 cout <<
" Static Friction: " << staticFCoeff << endl;
946 if (eContactType == ctBOGEY) {
947 cout <<
" Rolling Friction: " << rollingFCoeff << endl;
948 cout <<
" Steering Type: " << sSteerType[eSteerType] << endl;
949 cout <<
" Grouping: " << sBrakeGroup[eBrakeGrp] << endl;
950 cout <<
" Max Steer Angle: " << maxSteerAngle << endl;
951 cout <<
" Retractable: " << isRetractable << endl;
955 if (debug_lvl & 2 ) {
956 if (from == 0) cout <<
"Instantiated: FGLGear" << endl;
957 if (from == 1) cout <<
"Destroyed: FGLGear" << endl;
959 if (debug_lvl & 4 ) {
961 if (debug_lvl & 8 ) {
963 if (debug_lvl & 16) {
965 if (debug_lvl & 64) {
Element * FindElement(const std::string &el="")
Searches for a specified element.
FGColumnVector3 FindElementTripletConvertTo(const std::string &target_units)
Composes a 3-element column vector for the supplied location or orientation.
std::string GetAttributeValue(const std::string &key)
Retrieves an attribute.
std::string FindElementValue(const std::string &el="")
Searches for the named element and returns the string data belonging to it.
double FindElementValueAsNumberConvertTo(const std::string &el, const std::string &target_units)
Searches for the named element and converts and returns the data belonging to it.
Element * FindNextElement(const std::string &el="")
Searches for the next element as specified.
double GetDataAsNumber(void)
Converts the element data to a number.
double FindElementValueAsNumber(const std::string &el="")
Searches for the named element and returns the data belonging to it as a number.
This class implements a 3 element column vector.
double Magnitude(void) const
Length of the vector.
FGColumnVector3 & Normalize(void)
Normalize.
Encapsulates the JSBSim simulation executive.
std::shared_ptr< FGGroundReactions > GetGroundReactions(void) const
Returns the FGGroundReactions pointer.
bool IntegrationSuspended(void) const
Returns the simulation suspension state.
double GetSimTime(void) const
Returns the cumulative simulation time in seconds.
std::shared_ptr< FGInertial > GetInertial(void) const
Returns the FGInertial pointer.
std::shared_ptr< FGPropertyManager > GetPropertyManager(void) const
Returns a pointer to the property manager object.
Utility class that aids in the conversion of forces between coordinate systems and calculation of mom...
Represents a mathematical function.
double GetValue(void) const override
Retrieves the value of the function object.
static constexpr double Constrain(double min, double value, double max)
Constrain a value between a minimum and a maximum value.
static constexpr double RankineToCelsius(double rankine)
Converts from degrees Rankine to degrees Celsius.
const FGColumnVector3 & GetBodyForces(void) override
The Force vector for this gear.
FGLGear(Element *el, FGFDMExec *Executive, int number, const struct Inputs &input)
Constructor.
FGLocation holds an arbitrary location in the Earth centered Earth fixed reference frame (ECEF).
FGLocation LocalToLocation(const FGColumnVector3 &lvec) const
Conversion from Local frame coordinates to a location in the earth centered and fixed frame.
FGMatrix33 Transposed(void) const
Transposed matrix.
Models the Quaternion representation of rotations.
const FGMatrix33 & GetT(void) const
Transformation matrix.
bool GetSolid(void)
Gets the surface is a solid flag value.
double GetBumpiness(void)
Gets the normalized bumpiness factor associated with the surface.
double GetRollingFFactor(void)
Gets the rolling friction factor of the surface area.
double GetMaximumForce(void)
Gets the maximum force of the surface area.
double GetBumpHeight()
Returns the height of the bump at the provided offset.
double GetStaticFFactor(void)
Gets the static friction factor of the surface area.
double GetValue(void) const
Get the current table value.