//----------------------------------------------------------------------------- // Copyright (c) 2013 GarageGames, LLC // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to // deal in the Software without restriction, including without limitation the // rights to use, copy, modify, merge, publish, distribute, sublicense, and/or // sell copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS // IN THE SOFTWARE. //----------------------------------------------------------------------------- #ifndef _ATTRACTOR_CONTROLLER_H_ #include "2d/controllers/PointForceController.h" #endif // Script bindings. #include "PointForceController_ScriptBinding.h" //------------------------------------------------------------------------------ IMPLEMENT_CONOBJECT(PointForceController); //------------------------------------------------------------------------------ PointForceController::PointForceController() { // Reset he controller. mPosition.SetZero(); mRadius = 1.0f; mForce = 0.0f; } //------------------------------------------------------------------------------ PointForceController::~PointForceController() { } //------------------------------------------------------------------------------ void PointForceController::initPersistFields() { // Call parent. Parent::initPersistFields(); // Force. addProtectedField("Position", TypeVector2, Offset( mPosition, PointForceController), &defaultProtectedSetFn, &defaultProtectedGetFn, "The position of the attractor controller."); addProtectedField("Radius", TypeF32, Offset( mRadius, PointForceController), &defaultProtectedSetFn, &defaultProtectedGetFn, "The radius of the attractor circle centered on the attractors position."); addProtectedField("Force", TypeF32, Offset( mForce, PointForceController), &defaultProtectedSetFn, &defaultProtectedGetFn, "The force to apply to attact to the controller position."); } //------------------------------------------------------------------------------ void PointForceController::copyTo(SimObject* object) { // Call to parent. Parent::copyTo(object); // Cast to controller. PointForceController* pController = static_cast(object); // Sanity! AssertFatal(pController != NULL, "PointForceController::copyTo() - Object is not the correct type."); // Copy state. pController->setForce( getForce() ); } //------------------------------------------------------------------------------ void PointForceController::integrate( Scene* pScene, const F32 totalTime, const F32 elapsedTime, DebugStats* pDebugStats ) { // Finish if the attractor would have no effect. if ( mIsZero( mForce ) || mIsZero( mRadius ) ) return; // Prepare query filter. WorldQuery* pWorldQuery = prepareQueryFilter( pScene ); // Calculate the AABB of the attractor. b2AABB aabb; aabb.lowerBound.Set( mPosition.x - mRadius, mPosition.y - mRadius ); aabb.upperBound.Set( mPosition.x + mRadius, mPosition.y + mRadius ); // Query for candidate objects. pWorldQuery->anyQueryArea( aabb ); // Fetch results. typeWorldQueryResultVector& queryResults = pWorldQuery->getQueryResults(); // Iterate the results. for ( U32 n = 0; n < (U32)queryResults.size(); n++ ) { // Fetch the scene object. SceneObject* pSceneObject = queryResults[n].mpSceneObject; // Ignore if it's a static body. if ( pSceneObject->getBodyType() == b2BodyType::b2_staticBody ) continue; // Calculate the force direction to the controllers position. Vector2 forceDirection = mPosition - pSceneObject->getPosition(); // Skip if the position is outside the radius. if ( forceDirection.Length() > mRadius ) continue; // Normalize to the specified force. forceDirection.Normalize( mForce ); // Apply the force. pSceneObject->applyForce( forceDirection, true ); } } //------------------------------------------------------------------------------ void PointForceController::renderOverlay( Scene* pScene, const SceneRenderState* pSceneRenderState, BatchRender* pBatchRenderer ) { // Call parent. Parent::renderOverlay( pScene, pSceneRenderState, pBatchRenderer ); // Draw force radius. pScene->mDebugDraw.DrawCircle( mPosition, mRadius, ColorF(1.0f, 1.0f, 0.0f ) ); }