summaryrefslogblamecommitdiff
path: root/src/animation.cpp
blob: 129ab4bff3305e1d97052d340e9182fe70ce27c7 (plain) (tree)




















































                                                                             

                                                  
         
                           

















                                                         
                                    
















                                                                             











                                                                            


                  














































                                                                                
                                                                    







































































                                                                                   
 






































































































                                                                                                     














                                                                
                                           




                                
                                                                    

                                                             







                                                                





















































                                                                          
/*
 *  The Mana World
 *  Copyright 2004 The Mana World Development Team
 *
 *  This file is part of The Mana World.
 *
 *  The Mana World 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
 *  any later version.
 *
 *  The Mana World 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 The Mana World; if not, write to the Free Software
 *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 *
 *  $Id$
 */

#include "animation.h"

#include <libxml/tree.h>

#include "log.h"

#include "graphic/spriteset.h"

#include "resources/image.h"
#include "resources/resourcemanager.h"

#define READ_PROP(node, prop, name, target, cast) \
        prop = xmlGetProp(node, BAD_CAST name); \
        if (prop) { \
            target = cast((const char*)prop); \
            xmlFree(prop); \
        }

Animation::Animation():
    mTime(0)
{
    iCurrentPhase = mAnimationPhases.begin();
}

void
Animation::update(unsigned int time)
{
    mTime += time;
    if (!mAnimationPhases.empty())
    {
        unsigned int delay = iCurrentPhase->delay;
        while (mTime > delay && delay > 0)
        {
            mTime -= delay;
            iCurrentPhase++;
            if (iCurrentPhase == mAnimationPhases.end())
            {
                iCurrentPhase = mAnimationPhases.begin();
            }
        }
    }
}

int
Animation::getCurrentPhase()
{
    if (mAnimationPhases.empty())
    {
        return -1;
    }
    else
    {
        return iCurrentPhase->image;
    }
}

void
Animation::addPhase (int image, unsigned int delay, int offsetX, int offsetY)
{
    //add new phase to animation list
    AnimationPhase newPhase;
    newPhase.image = image;
    newPhase.delay = delay;
    newPhase.offsetX = offsetX;
    newPhase.offsetY = offsetY;
    mAnimationPhases.push_back(newPhase);
    //reset animation circle
    iCurrentPhase = mAnimationPhases.begin();
}

int
Animation::getLength()
{
    std::list<AnimationPhase>::iterator i;
    int length = 0;
    if (!mAnimationPhases.empty())
    {
        for (i = mAnimationPhases.begin(); i != mAnimationPhases.end(); i++)
        {
            length += (*i).delay;
        }
    }
    return length;
}

Action::Action()
    :mImageset("")
{
    //NOOP
}

Action::~Action()
{
    for (AnimationIterator i = mAnimations.begin(); i != mAnimations.end(); i++)
    {
        delete i->second;
    }
    mAnimations.clear();
}

Animation *
Action::getAnimation(std::string direction)
{
    Animation *animation = NULL;
    AnimationIterator i = mAnimations.find(direction);
    if (i == mAnimations.end())
    {
        //when the direction isn't defined just use another one
        animation = mAnimations["default"];
    }
    else
    {
        animation = mAnimations[direction];
    }

    return animation;
}

void
Action::setAnimation(std::string direction, Animation *animation)
{
    //set first direction as default direction
    if (mAnimations.empty())
    {
        mAnimations["default"] = animation;
    }

    mAnimations[direction] = animation;
}


AnimatedSprite::AnimatedSprite(std::string animationFile, int variant):
    mAction("stand"), mDirection("down"), mLastTime(0), mSpeed(1.0f)
{
    int variant_num = 0;
    int variant_offset = 0;
    int size;
    ResourceManager *resman = ResourceManager::getInstance();
    char *data = (char*)resman->loadFile(
            animationFile.c_str(), size);

    if (!data) {
        logger->error("Animation: Could not find " + animationFile + " !");
    }

    xmlDocPtr doc = xmlParseMemory(data, size);
    free(data);

    if (!doc)
    {
        logger->error("Animation: Error while parsing animation definition file!");
        return;
    }

    xmlNodePtr node = xmlDocGetRootElement(doc);
    if (!node || !xmlStrEqual(node->name, BAD_CAST "sprite"))
    {
        logger->error("Animation: this is not a valid animation definition file!");
        return;
    }

    //get the variant
    xmlChar *prop = NULL;
    READ_PROP(node, prop, "variants", variant_num, atoi);
    READ_PROP(node, prop, "variant_offset", variant_offset, atoi);

    if (variant_num > 0 && variant < variant_num )
    {
        variant_offset *= variant;
    }
    else
    {
        variant_offset = 0;
    }

    for (node = node->xmlChildrenNode; node != NULL; node = node->next)
    {
        if (xmlStrEqual(node->name, BAD_CAST "imageset"))
        {
            int width = 0, height = 0;
            std::string name = "", imageSrc = "";
            xmlChar *prop = NULL;
            READ_PROP(node, prop, "name", name, );
            READ_PROP(node, prop, "src", imageSrc, );
            READ_PROP(node, prop, "width", width, atoi);
            READ_PROP(node, prop, "height", height, atoi);

            Spriteset *spriteset = resman->createSpriteset(
                    imageSrc, width, height);
            if (!spriteset)
            {
                logger->error("Couldn't load spriteset!");
            }
            else
            {
                mSpritesets[name] = spriteset;
            }
        }
        // get action
        else if (xmlStrEqual(node->name, BAD_CAST "action"))
        {
            std::string name = "", imageset = "";
            xmlChar *prop = NULL;
            READ_PROP(node, prop, "name", name, );
            READ_PROP(node, prop, "imageset", imageset, );

            Action *action = new Action();
            mActions[name] = action;
            action->setImageset(imageset);

            //get animations
            for (   xmlNodePtr animationNode = node->xmlChildrenNode;
                    animationNode != NULL;
                    animationNode = animationNode->next)
            {
                if (xmlStrEqual(animationNode->name, BAD_CAST "animation"))
                {
                    std::string direction = "";


                    Animation *animation = new Animation();
                    READ_PROP(animationNode, prop, "direction", direction, );

                    //get animation phases
                    for (   xmlNodePtr phaseNode = animationNode->xmlChildrenNode;
                            phaseNode != NULL;
                            phaseNode = phaseNode->next)
                    {
                        int index = -1;
                        int start = 0;
                        int end = 0;
                        int delay = 0;
                        int offsetX = 0;
                        int offsetY = 0;
                        if (xmlStrEqual(phaseNode->name, BAD_CAST "frame"))
                        {
                            READ_PROP(phaseNode, prop, "index", index, atoi);
                            READ_PROP(phaseNode, prop, "delay", delay, atoi);
                            READ_PROP(phaseNode, prop, "offsetX", offsetX, atoi);
                            READ_PROP(phaseNode, prop, "offsetY", offsetY, atoi);
                            offsetY = offsetY - mSpritesets[imageset]->getHeight() + 32;
                            offsetX = offsetX - mSpritesets[imageset]->getWidth() / 2 + 16;
                            animation->addPhase(index + variant_offset, delay, offsetX, offsetY);
                        }
                        if (xmlStrEqual(phaseNode->name, BAD_CAST "sequence"))
                        {
                            READ_PROP(phaseNode, prop, "start", start, atoi);
                            READ_PROP(phaseNode, prop, "end", end, atoi);
                            READ_PROP(phaseNode, prop, "delay", delay, atoi);
                            offsetY = 0 - mSpritesets[imageset]->getHeight() + 32;
                            offsetX = 0 - mSpritesets[imageset]->getWidth() / 2 + 16;
                            while (end >= start)
                            {
                                animation->addPhase(start + variant_offset, delay, offsetX, offsetY);
                                start++;
                            }
                        }
                    } // for phaseNode
                    action->setAnimation(direction, animation);
                } // if "<animation>"
            } // for animationNode
        } // if "<imageset>" else if "<action>"
    } // for node

    //complete missing actions
    substituteAction("stand", "");
    substituteAction("walk", "stand");
    substituteAction("walk", "run");
    substituteAction("attack", "stand");
    substituteAction("attack_swing", "attack");
    substituteAction("attack_stab", "attack_swing");
    substituteAction("attack_bow", "attack_stab");
    substituteAction("attack_throw", "attack_swing");
    substituteAction("cast_magic", "attack_swing");
    substituteAction("use_item", "cast_magic");
    substituteAction("sit", "stand");
    substituteAction("sleeping", "sit");
    substituteAction("hurt", "stand");
    substituteAction("dead", "hurt");

    xmlFreeDoc(doc);
}

void
AnimatedSprite::substituteAction(std::string complete, std::string with)
{
    if (mActions.find(complete) == mActions.end())
    {
        mActions[complete] = mActions[with];
    }
}

AnimatedSprite::~AnimatedSprite()
{
    for (SpritesetIterator i = mSpritesets.begin(); i != mSpritesets.end(); i++)
    {
        delete i->second;
    }
    mSpritesets.clear();
}

void
AnimatedSprite::play(std::string action)
{
    if (mAction != action)
    {
        mAction = action;
    }
    mLastTime = 0;
    mSpeed = 1.0f;
}

void
AnimatedSprite::play(std::string action, int time)
{
    if (mAction != action)
    {
        mAction = action;
    }
    mLastTime = 0;
    int animationLength = 0;
    Action *nextAction = mActions[mAction];
    Animation *animation = nextAction->getAnimation(mDirection);
    animationLength = animation->getLength();
    mSpeed = (float)animationLength / time;
}

void
AnimatedSprite::update(int time)
{
    // avoid freaking out at first frame or when tick_time overflows
    if (time < mLastTime || mLastTime == 0) mLastTime = time;

    // if not enough time have passed yet, do nothing
    if (time > mLastTime)
    {
        Action *action = mActions[mAction];
        Animation *animation = action->getAnimation(mDirection);
        animation->update((time - mLastTime) * mSpeed, mSpeed);
        mLastTime = time;
    }
}

bool
AnimatedSprite::draw(Graphics * graphics, Sint32 posX, Sint32 posY)
{
    Sint32 offsetX, offsetY;
    Action *action = mActions[mAction];
    Spriteset *spriteset = mSpritesets[action->getImageset()];
    Animation *animation = action->getAnimation(mDirection);

    if (animation->getCurrentPhase() >= 0)
    {
        Image *image = spriteset->get(animation->getCurrentPhase());
        offsetX = animation->getOffsetX();
        offsetY = animation->getOffsetY();
        return graphics->drawImage(image, posX + offsetX, posY + offsetY);
    }
    else
    {
        return false;
    }
}

Image *
AnimatedSprite::getCurrentFrame()
{
    Action *action = mActions[mAction];
    Spriteset *spriteset = mSpritesets[action->getImageset()];
    Animation *animation = action->getAnimation(mDirection);
    if (animation->getCurrentPhase() >= 0)
    {
        return spriteset->get(animation->getCurrentPhase());
    }
    else
    {
        return NULL;
    }
}

int
AnimatedSprite::getWidth()
{
    Action *action = mActions[mAction];
    Spriteset *spriteset = mSpritesets[action->getImageset()];
    return spriteset->getWidth();
}

int
AnimatedSprite::getHeight()
{
    Action *action = mActions[mAction];
    Spriteset *spriteset = mSpritesets[action->getImageset()];
    return spriteset->getHeight();
}