/* This file is part of StepCore library. Copyright (C) 2007 Vladimir Kuznetsov StepCore library 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 2 of the License, or (at your option) any later version. StepCore library 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 StepCore; if not, write to the Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */ #include "gravitation.h" #include "particle.h" #include "rigidbody.h" #include #include namespace StepCore { STEPCORE_META_OBJECT(GravitationForce, QT_TRANSLATE_NOOP("ObjectClass", "GravitationForce"), QT_TR_NOOP("Gravitation force"), 0, STEPCORE_SUPER_CLASS(Item) STEPCORE_SUPER_CLASS(Force), STEPCORE_PROPERTY_RW(double, gravitationConst, QT_TRANSLATE_NOOP("PropertyName", "gravitationConst"), STEPCORE_FROM_UTF8(QT_TRANSLATE_NOOP("Units", "N m²/kg²")), QT_TR_NOOP("Gravitation constant"), gravitationConst, setGravitationConst)) STEPCORE_META_OBJECT(GravitationForceErrors, QT_TRANSLATE_NOOP("ObjectClass", "GravitationForceErrors"), QT_TR_NOOP("Errors class for GravitationForce"), 0, STEPCORE_SUPER_CLASS(ObjectErrors), STEPCORE_PROPERTY_RW(double, gravitationConstVariance, QT_TRANSLATE_NOOP("PropertyName", "gravitationConstVariance"), STEPCORE_FROM_UTF8(QT_TRANSLATE_NOOP("Units", "N m²/kg²")), QT_TR_NOOP("Gravitation constant variance"), gravitationConstVariance, setGravitationConstVariance)) STEPCORE_META_OBJECT(WeightForce, QT_TRANSLATE_NOOP("ObjectClass", "WeightForce"), QT_TR_NOOP("Weight force"), 0, STEPCORE_SUPER_CLASS(Item) STEPCORE_SUPER_CLASS(Force), STEPCORE_PROPERTY_RW(double, weightConst, QT_TRANSLATE_NOOP("PropertyName", "weightConst"), STEPCORE_FROM_UTF8(QT_TRANSLATE_NOOP("Units", "m/s²")), QT_TR_NOOP("Weight constant"), weightConst, setWeightConst)) STEPCORE_META_OBJECT(WeightForceErrors, QT_TRANSLATE_NOOP("ObjectClass", "WeightForceErrors"), QT_TR_NOOP("Errors class for WeightForce"), 0, STEPCORE_SUPER_CLASS(ObjectErrors), STEPCORE_PROPERTY_RW(double, weightConstVariance, QT_TRANSLATE_NOOP("PropertyName", "weightConstVariance"), STEPCORE_FROM_UTF8(QT_TRANSLATE_NOOP("Units", "m/s²")), QT_TR_NOOP("Weight constant variance"), weightConstVariance, setWeightConstVariance)) GravitationForce* GravitationForceErrors::gravitationForce() const { return static_cast(owner()); } WeightForce* WeightForceErrors::weightForce() const { return static_cast(owner()); } GravitationForce::GravitationForce(double gravitationConst) : _gravitationConst(gravitationConst) { gravitationForceErrors()->setGravitationConstVariance( square(Constants::GravitationalError)); } void GravitationForce::calcForce(bool calcVariances) { const BodyList::const_iterator end = world()->bodies().end(); for(BodyList::const_iterator b1 = world()->bodies().begin(); b1 != end; ++b1) { if(!(*b1)->metaObject()->inherits()) continue; for(BodyList::const_iterator b2 = b1+1; b2 != end; ++b2) { if(!(*b2)->metaObject()->inherits()) continue; Particle* p1 = static_cast(*b1); Particle* p2 = static_cast(*b2); Vector2d r = p2->position() - p1->position(); double rnorm2 = r.squaredNorm(); Vector2d force = _gravitationConst * p1->mass() * p2->mass() * r / (rnorm2*sqrt(rnorm2)); p1->applyForce(force); force = -force; p2->applyForce(force); if(calcVariances) { // XXX: CHECKME ParticleErrors* pe1 = p1->particleErrors(); ParticleErrors* pe2 = p2->particleErrors(); Vector2d rV = pe2->positionVariance() + pe1->positionVariance(); Vector2d forceV = force.cwise().square() .cwise()* ( Vector2d(gravitationForceErrors()->_gravitationConstVariance / square(_gravitationConst) + pe1->massVariance() / square(p1->mass()) + pe2->massVariance() / square(p2->mass())) + Vector2d(rV[0] * square(1/r[0] - 3*r[0]/rnorm2) + rV[1] * square(3*r[1]/rnorm2), rV[1] * square(1/r[1] - 3*r[1]/rnorm2) + rV[0] * square(3*r[0]/rnorm2))); pe1->applyForceVariance(forceV); pe2->applyForceVariance(forceV); } } } } WeightForce::WeightForce(double weightConst) : _weightConst(weightConst) { weightForceErrors()->setWeightConstVariance( square(Constants::WeightAccelError)); } void WeightForce::calcForce(bool calcVariances) { Vector2d g(0., -_weightConst); const BodyList::const_iterator end = world()->bodies().end(); for(BodyList::const_iterator b1 = world()->bodies().begin(); b1 != end; ++b1) { if((*b1)->metaObject()->inherits()) { Particle* p1 = static_cast(*b1); p1->applyForce(g*p1->mass()); if(calcVariances) { ParticleErrors* pe1 = p1->particleErrors(); Vector2d forceV(0., square(_weightConst)*pe1->massVariance()+ square(p1->mass())*weightForceErrors()->weightConstVariance()); pe1->applyForceVariance(forceV); } } else if((*b1)->metaObject()->inherits()) { RigidBody* rb1 = static_cast(*b1); rb1->applyForce(g*rb1->mass(), rb1->position()); if(calcVariances) { RigidBodyErrors* rbe1 = rb1->rigidBodyErrors(); Vector2d forceV(0., square(_weightConst)*rbe1->massVariance()+ square(rb1->mass())*weightForceErrors()->weightConstVariance()); rbe1->applyForceVariance(g*rb1->mass(), rb1->position(), forceV, rbe1->positionVariance()); } } } } } // namespace StepCore