/* Copyright (C) 2007, 2010 - Bit-Blot This file is part of Aquaria. Aquaria 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. This program 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 this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ #ifndef SKELETALSPRITE_H #define SKELETALSPRITE_H #include "Quad.h" #include "SimpleIStringStream.h" #include #include "Interpolators.h" // for 2d system only enum AnimationCommand { AC_PRT_LOAD =0, AC_PRT_START , AC_PRT_STOP , AC_SEGS_START , AC_FRM_SHOW , AC_SND_PLAY , AC_SEGS_STOP, AC_SET_PASS, AC_RESET_PASS, AC_SET_FH }; class ParticleEffect; class SkeletalSprite; class Bone : public CollideQuad { friend class SkeletalSprite; public: Bone(); virtual ~Bone(); void setAnimated(int a); enum { ANIM_NONE = 0x00, ANIM_POS = 0x01, ANIM_ROT = 0x02, ANIM_ALL = ANIM_POS | ANIM_ROT }; void createStrip(bool vert, int num); void addFrame(const std::string &gfx); void showFrame(int i); void destroy() OVERRIDE; std::string gfx; std::string name; size_t boneIdx; int pidx, rbp; std::string prt; std::vector changeStrip; bool generateCollisionMask; bool enableCollision; int animated; Vector originalScale; bool canCollide() const; void addSegment(Bone *b); ParticleEffect *getEmitter(unsigned slot) const; int segmentChain; void updateSegments(); void updateSegment(Bone *b, const Vector &diff); SkeletalSprite *skeleton; void setSegmentProps(int minDist, int maxDist, bool reverse); Vector segmentOffset; bool stripVert; bool fileRenderQuad; bool selectable; bool originalFH; bool inheritPass; int originalRenderPass; // stores the render pass originally set in the XML file. For AC_RESET_PASS. void spawnParticlesFromCollisionMask(const char *p, unsigned intv, int layer, float rotz = 0); Vector getCollisionMaskNormal(Vector pos, float dist) const; virtual void renderCollision(const RenderState& rs) const OVERRIDE; protected: std::vector emitters; int minDist, maxDist, reverse; std::vector segments; void onUpdate(float dt); public: std::vector collisionMask; std::vector transformedCollisionMask; float collisionMaskRadius; std::vector framegfx; }; class BoneCommand { public: bool parse(Bone *b, SimpleIStringStream &is); void run(); AnimationCommand command; Bone *b; int slot; std::string file; }; class BoneKeyframe { public: BoneKeyframe() : idx(0), x(0), y(0), rot(0), sx(1), sy(1), doScale(0) {} size_t idx; int x, y, rot; float sx, sy; bool doScale; std::vector grid; std::vector controlpoints; }; class SkeletalKeyframe { public: SkeletalKeyframe() { lerpType = 0; t = 0; } int lerpType; float t; std::string sound; std::vector keyframes; BoneKeyframe *getBoneKeyframe(size_t idx); std::string cmd; std::vector commands; void copyAllButTime(SkeletalKeyframe *copy); }; class BoneGridInterpolator { public: size_t idx; BSpline2D bsp; bool storeBoneByIdx; void updateGridOnly(BoneKeyframe& bk, const Bone *bone); void updateGridAndBone(BoneKeyframe& bk, Bone *bone); }; class Animation { public: Animation(); std::string name; typedef std::vector Keyframes; Keyframes keyframes; SkeletalKeyframe *getKeyframe(size_t key); SkeletalKeyframe *getLastKeyframe(); SkeletalKeyframe *getFirstKeyframe(); size_t getKeyframeIndexBefore(float t) const; // last anim that's <= t void cloneKey(size_t key, float toffset); void deleteKey(size_t key); void reorderKeyframes(); float getAnimationLength(); size_t getSkeletalKeyframeIndex(SkeletalKeyframe *skey); size_t getNumKeyframes() const; void reverse(); bool resetOnEnd; BoneGridInterpolator *getBoneGridInterpolator(size_t boneIdx); typedef std::vector Interpolators; Interpolators interpolators; }; class SkeletalSprite; class AnimationLayer { public: //---- AnimationLayer(); void setSkeletalSprite(SkeletalSprite *s); Animation *getCurrentAnimation(); Animation *getCurrentAnimationOrNull(); void animate(const std::string &animation, int loop); void update(float dt); void updateBones(); void keyframeReached(SkeletalKeyframe *k, size_t idx); void stopAnimation(); float getAnimationLength(); void createTransitionAnimation(Animation& to, float time); void playAnimation(int idx, int loop); void playCurrentAnimation(int loop); void enqueueAnimation(const std::string& anim, int loop); float transitionAnimate(std::string anim, float time, int loop); void setTimeMultiplier(float t); bool isAnimating() const; bool contains(const Bone *b) const; void resetPass(); void setTimer(float t, bool runKeyframes); //---- float fallThru; float fallThruSpeed; std::string name; std::vector ignoreBones; std::vector includeBones; SkeletalSprite *s; int lastKeyframeIndex; float timer; int loop; Animation blendAnimation; std::string enqueuedAnimation; int enqueuedAnimationLoop; //float timeMultiplier; //HACK: should be a lerped float InterpolatedVector timeMultiplier; float animationLength; size_t currentAnimation; bool animating; }; class SkeletalSprite : public RenderObject { public: SkeletalSprite(); virtual ~SkeletalSprite(); virtual void destroy(); void loadSkeletal(const std::string &fn); bool saveSkeletal(const std::string &fn); void loadSkin(const std::string &fn); Bone *getBoneByIdx(size_t idx); Bone *getBoneByName(const std::string &name); void animate(const std::string &animation, int loop = 0, int layer=0); void setTime(float time, size_t layer=0); void updateBones(); void playCurrentAnimation(int loop=0, int layer=0); void stopAnimation(int layer=0); void stopAllAnimations(); float transitionAnimate(const std::string& anim, float time, int loop=0, int layer=0); bool isAnimating(size_t layer=0) const; void setTimeMultiplier(float t, int layer=0); int findSelectableBoneIdxClosestTo(const Vector& pos, bool mustBeInBox = true) const; // -1 if none found, otherwise bones[idx] to get the bone Animation *getCurrentAnimation(size_t layer=0); Animation *getCurrentAnimationOrNull(size_t layer=0); void nextAnimation(); void prevAnimation(); void lastAnimation(); void firstAnimation(); bool selectAnimation(const char *name); void setFreeze(bool f); Animation *getAnimation(const std::string& anim); std::vector animations; std::vector bones; bool isLoaded(); size_t getNumAnimLayers() const { return animLayers.size(); } AnimationLayer* getAnimationLayer(size_t l); size_t getBoneIdx(Bone *b); void toggleBone(size_t idx, int v); void setAnimationKeyNotify(RenderObject *r); std::string filenameLoaded; static std::string animationPath, skinPath, secondaryAnimationPath; static void clearCache(); protected: bool frozen; RenderObject *animKeyNotify; bool loaded; friend class AnimationLayer; std::vector animLayers; Bone* initBone(int idx, std::string gfx, int pidx, bool rbp=false, std::string name="", float cr=0, bool fh=false, bool fv=false); void deleteBones(); void onUpdate(float dt); }; #endif