/** * @file llmotion.h * @brief Implementation of LLMotion class. * * $LicenseInfo:firstyear=2001&license=viewerlgpl$ * Second Life Viewer Source Code * Copyright (C) 2010, Linden Research, Inc. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; * version 2.1 of the License only. * * This 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA * * Linden Research, Inc., 945 Battery Street, San Francisco, CA 94111 USA * $/LicenseInfo$ */ #ifndef LL_LLMOTION_H #define LL_LLMOTION_H //----------------------------------------------------------------------------- // Header files //----------------------------------------------------------------------------- #include <string> #include "llerror.h" #include "llpose.h" #include "lluuid.h" class LLCharacter; //----------------------------------------------------------------------------- // class LLMotion //----------------------------------------------------------------------------- class LLMotion { friend class LLMotionController; public: enum LLMotionBlendType { NORMAL_BLEND, ADDITIVE_BLEND }; enum LLMotionInitStatus { STATUS_FAILURE, STATUS_SUCCESS, STATUS_HOLD }; // Constructor LLMotion(const LLUUID &id); // Destructor virtual ~LLMotion(); public: //------------------------------------------------------------------------- // functions to support MotionController and MotionRegistry //------------------------------------------------------------------------- // get the name of this instance const std::string &getName() const { return mName; } // set the name of this instance void setName(const std::string &name) { mName = name; } const LLUUID& getID() const { return mID; } // returns the pose associated with the current state of this motion virtual LLPose* getPose() { return &mPose;} void fadeOut(); void fadeIn(); F32 getFadeWeight() const { return mFadeWeight; } F32 getStopTime() const { return mStopTimestamp; } virtual void setStopTime(F32 time); BOOL isStopped() const { return mStopped; } void setStopped(BOOL stopped) { mStopped = stopped; } BOOL isBlending(); // Activation functions. // It is OK for other classes to activate a motion, // but only the controller can deactivate it. // Thus, if mActive == TRUE, the motion *may* be on the controllers active list, // but if mActive == FALSE, the motion is gauranteed not to be on the active list. protected: // Used by LLMotionController only void deactivate(); BOOL isActive() { return mActive; } public: void activate(F32 time); public: //------------------------------------------------------------------------- // animation callbacks to be implemented by subclasses //------------------------------------------------------------------------- // motions must specify whether or not they loop virtual BOOL getLoop() = 0; // motions must report their total duration virtual F32 getDuration() = 0; // motions must report their "ease in" duration virtual F32 getEaseInDuration() = 0; // motions must report their "ease out" duration. virtual F32 getEaseOutDuration() = 0; // motions must report their priority level virtual LLJoint::JointPriority getPriority() = 0; // motions must report their blend type virtual LLMotionBlendType getBlendType() = 0; // called to determine when a motion should be activated/deactivated based on avatar pixel coverage virtual F32 getMinPixelArea() = 0; // run-time (post constructor) initialization, // called after parameters have been set // must return true to indicate success and be available for activation virtual LLMotionInitStatus onInitialize(LLCharacter *character) = 0; // called per time step // must return TRUE while it is active, and // must return FALSE when the motion is completed. virtual BOOL onUpdate(F32 activeTime, U8* joint_mask) = 0; // called when a motion is deactivated virtual void onDeactivate() = 0; // can we crossfade this motion with a new instance when restarted? // should ultimately always be TRUE, but lack of emote blending, etc // requires this virtual BOOL canDeprecate(); // optional callback routine called when animation deactivated. void setDeactivateCallback( void (*cb)(void *), void* userdata ); protected: // called when a motion is activated // must return TRUE to indicate success, or else // it will be deactivated virtual BOOL onActivate() = 0; void addJointState(const LLPointer<LLJointState>& jointState); protected: LLPose mPose; BOOL mStopped; // motion has been stopped; BOOL mActive; // motion is on active list (can be stopped or not stopped) //------------------------------------------------------------------------- // these are set implicitly by the motion controller and // may be referenced (read only) in the above handlers. //------------------------------------------------------------------------- std::string mName; // instance name assigned by motion controller LLUUID mID; F32 mActivationTimestamp; // time when motion was activated F32 mStopTimestamp; // time when motion was told to stop F32 mSendStopTimestamp; // time when simulator should be told to stop this motion F32 mResidualWeight; // blend weight at beginning of stop motion phase F32 mFadeWeight; // for fading in and out based on LOD U8 mJointSignature[3][LL_CHARACTER_MAX_ANIMATED_JOINTS]; // signature of which joints are animated at what priority void (*mDeactivateCallback)(void* data); void* mDeactivateCallbackUserData; }; //----------------------------------------------------------------------------- // LLTestMotion //----------------------------------------------------------------------------- class LLTestMotion : public LLMotion { public: LLTestMotion(const LLUUID &id) : LLMotion(id){} ~LLTestMotion() {} static LLMotion *create(const LLUUID& id) { return new LLTestMotion(id); } BOOL getLoop() { return FALSE; } F32 getDuration() { return 0.0f; } F32 getEaseInDuration() { return 0.0f; } F32 getEaseOutDuration() { return 0.0f; } LLJoint::JointPriority getPriority() { return LLJoint::HIGH_PRIORITY; } LLMotionBlendType getBlendType() { return NORMAL_BLEND; } F32 getMinPixelArea() { return 0.f; } LLMotionInitStatus onInitialize(LLCharacter*) { LL_INFOS() << "LLTestMotion::onInitialize()" << LL_ENDL; return STATUS_SUCCESS; } BOOL onActivate() { LL_INFOS() << "LLTestMotion::onActivate()" << LL_ENDL; return TRUE; } BOOL onUpdate(F32 time, U8* joint_mask) { LL_INFOS() << "LLTestMotion::onUpdate(" << time << ")" << LL_ENDL; return TRUE; } void onDeactivate() { LL_INFOS() << "LLTestMotion::onDeactivate()" << LL_ENDL; } }; //----------------------------------------------------------------------------- // LLNullMotion //----------------------------------------------------------------------------- class LLNullMotion : public LLMotion { public: LLNullMotion(const LLUUID &id) : LLMotion(id) {} ~LLNullMotion() {} static LLMotion *create(const LLUUID &id) { return new LLNullMotion(id); } // motions must specify whether or not they loop /*virtual*/ BOOL getLoop() { return TRUE; } // motions must report their total duration /*virtual*/ F32 getDuration() { return 1.f; } // motions must report their "ease in" duration /*virtual*/ F32 getEaseInDuration() { return 0.f; } // motions must report their "ease out" duration. /*virtual*/ F32 getEaseOutDuration() { return 0.f; } // motions must report their priority level /*virtual*/ LLJoint::JointPriority getPriority() { return LLJoint::HIGH_PRIORITY; } // motions must report their blend type /*virtual*/ LLMotionBlendType getBlendType() { return NORMAL_BLEND; } // called to determine when a motion should be activated/deactivated based on avatar pixel coverage /*virtual*/ F32 getMinPixelArea() { return 0.f; } // run-time (post constructor) initialization, // called after parameters have been set // must return true to indicate success and be available for activation /*virtual*/ LLMotionInitStatus onInitialize(LLCharacter *character) { return STATUS_SUCCESS; } // called when a motion is activated // must return TRUE to indicate success, or else // it will be deactivated /*virtual*/ BOOL onActivate() { return TRUE; } // called per time step // must return TRUE while it is active, and // must return FALSE when the motion is completed. /*virtual*/ BOOL onUpdate(F32 activeTime, U8* joint_mask) { return TRUE; } // called when a motion is deactivated /*virtual*/ void onDeactivate() {} }; #endif // LL_LLMOTION_H