summaryrefslogblamecommitdiff
path: root/src/net/beinghandler.cpp
blob: 51609b1dc7c81c60c324f7ca768db3c4da34e534 (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 "beinghandler.h"

#include <SDL_types.h>

#include "messagein.h"
#include "protocol.h"

#include "../being.h"
#include "../beingmanager.h"
#include "../game.h"
#include "../localplayer.h"
#include "../log.h"
#include "../main.h"
#include "../particle.h"
#include "../sound.h"

const int EMOTION_TIME = 150;    /**< Duration of emotion icon */

BeingHandler::BeingHandler()
{
    static const Uint16 _messages[] = {
        //SMSG_BEING_VISIBLE,
        //SMSG_BEING_MOVE,
        //SMSG_BEING_REMOVE,
        //SMSG_BEING_ACTION,
        //SMSG_BEING_LEVELUP,
        //SMSG_BEING_EMOTION,
        //SMSG_BEING_CHANGE_LOOKS,
        //SMSG_BEING_NAME_RESPONSE,
        //SMSG_PLAYER_UPDATE_1,
        //SMSG_PLAYER_UPDATE_2,
        //SMSG_PLAYER_MOVE,
        //0x0119,
        GPMSG_BEING_ATTACK,
        GPMSG_BEING_ENTER,
        GPMSG_BEING_LEAVE,
        GPMSG_BEINGS_MOVE,
        GPMSG_BEINGS_DAMAGE,
        GPMSG_BEING_ACTION_CHANGE,
        GPMSG_BEING_LOOKS_CHANGE,
        0
    };
    handledMessages = _messages;
}

void BeingHandler::handleMessage(MessageIn &msg)
{
    /*
    Uint32 id;
    Uint16 job, speed;
    Uint16 headBottom, headTop, headMid;
    Sint16 param1;
    Sint8 type;
    Being *srcBeing, *dstBeing;
    */

    switch (msg.getId())
    {
        case GPMSG_BEING_ENTER:
            handleBeingEnterMessage(msg);
            break;
        case GPMSG_BEING_LEAVE:
            handleBeingLeaveMessage(msg);
            break;
        case GPMSG_BEINGS_MOVE:
            handleBeingsMoveMessage(msg);
            break;
        case GPMSG_BEING_ATTACK:
            handleBeingAttackMessage(msg);
            break;
        case GPMSG_BEINGS_DAMAGE:
            handleBeingsDamageMessage(msg);
            break;
        case GPMSG_BEING_ACTION_CHANGE:
            handleBeingActionChangeMessage(msg);
            break;
        case GPMSG_BEING_LOOKS_CHANGE:
            handleBeingLooksChangeMessage(msg);
            break;

        /*
        case SMSG_BEING_VISIBLE:
        case SMSG_BEING_MOVE:
            // Information about a being in range
            id = msg.readLong();
            speed = msg.readShort();
            msg.readShort();  // unknown
            msg.readShort();  // unknown
            msg.readShort();  // option
            job = msg.readShort();  // class

            dstBeing = beingManager->findBeing(id);

            if (dstBeing == NULL)
            {
                // Being with id >= 110000000 and job 0 are better
                // known as ghosts, so don't create those.
                if (job == 0 && id >= 110000000)
                {
                    break;
                }

                dstBeing = beingManager->createBeing(id, job);
            }
            else if (msg.getId() == 0x0078)
            {
                dstBeing->clearPath();
                dstBeing->mFrame = 0;
                dstBeing->mWalkTime = tick_time;
                dstBeing->setAction(Being::STAND);
            }

            // Prevent division by 0 when calculating frame
            if (speed == 0) { speed = 150; }

            dstBeing->setWalkSpeed(speed);
            dstBeing->mJob = job;
            dstBeing->setHairStyle(msg.readShort());
            dstBeing->setWeapon(msg.readShort());
            dstBeing->setVisibleEquipment(
                    Being::BOTTOMCLOTHES_SPRITE, msg.readShort());

            if (msg.getId() == SMSG_BEING_MOVE)
            {
                msg.readLong(); // server tick
            }

            msg.readShort();  // shield
            dstBeing->setVisibleEquipment(Being::HAIT_SPRITE, msg.readShort());
            dstBeing->setVisibleEquipment(
                    Being::TOPCLOTHES_SPRITE, msg.readShort());
            dstBeing->setHairColor(msg.readShort());
            msg.readShort();  // unknown
            msg.readShort();  // head dir
            msg.readShort();  // guild
            msg.readShort();  // unknown
            msg.readShort();  // unknown
            msg.readShort();  // manner
            msg.readShort();  // karma
            msg.readByte();   // unknown
            dstBeing->setSex(1 - msg.readByte());   // sex

            if (msg.getId() == SMSG_BEING_MOVE)
            {
                //Uint16 srcX, srcY, dstX, dstY;
                //msg.readCoordinatePair(srcX, srcY, dstX, dstY);
                //dstBeing->setAction(Being::STAND);
                //dstBeing->mX = srcX;
                //dstBeing->mY = srcY;
                //dstBeing->setDestination(dstX, dstY);
            }
            else
            {
                //Uint8 dir;
                //msg->readCoordinates(dstBeing->mX, dstBeing->mY, dir);
                //dstBeing->setDirection(dir);
            }

            msg.readByte();   // unknown
            msg.readByte();   // unknown
            msg.readByte();   // unknown / sit
            break;

        case SMSG_BEING_REMOVE:
            // A being should be removed or has died
            dstBeing = beingManager->findBeing(msg.readLong());

            if (!dstBeing)
                break;

            if (msg.readByte() == 1)
            {
                dstBeing->setAction(Being::DEAD);
            }
            else
            {
                beingManager->destroyBeing(dstBeing);
            }

            if (dstBeing == player_node->getTarget())
            {
                player_node->stopAttack();
            }
            break;

        case SMSG_BEING_ACTION:
            srcBeing = beingManager->findBeing(msg.readLong());
            dstBeing = beingManager->findBeing(msg.readLong());
            msg.readLong();   // server tick
            msg.readLong();   // src speed
            msg.readLong();   // dst speed
            param1 = msg.readShort();
            msg.readShort();  // param 2
            type = msg.readByte();
            msg.readShort();  // param 3

            switch (type)
            {
                case 0: // Damage
                    if (dstBeing) {
                        dstBeing->takeDamage(param1);
                    }
                    if (srcBeing) {
                        srcBeing->handleAttack(dstBeing, param1);
                    }
                    break;

                case 2: // Sit
                    if (srcBeing) {
                        srcBeing->mFrame = 0;
                        srcBeing->setAction(Being::SIT);
                    }
                    break;

                case 3: // Stand up
                    if (srcBeing) {
                        srcBeing->mFrame = 0;
                        srcBeing->setAction(Being::STAND);
                    }
                    break;
            }
            break;

        case SMSG_BEING_LEVELUP:
            id = (Uint32) msg->readLong();

            if (id == player_node->getId()) {
                logger->log("Level up");
                sound.playSfx("sfx/levelup.ogg");
            }
            else {
                logger->log("Someone else went level up");
            }
            Particle *levelupFX;
            if (msg->readLong() == 0) { // type
                levelupFX = particleEngine->addEffect("graphics/particles/levelup.particle.xml", 0, 0);
            }
            else {
                levelupFX = particleEngine->addEffect("graphics/particles/skillup.particle.xml", 0, 0);
            }
            beingManager->findBeing(id)->controlParticle(levelupFX);
            break;

        case SMSG_BEING_EMOTION:
            if (!(dstBeing = beingManager->findBeing(msg.readLong())))
            {
                break;
            }

            dstBeing->mEmotion = msg.readByte();
            dstBeing->mEmotionTime = EMOTION_TIME;
            break;

        case SMSG_BEING_CHANGE_LOOKS:
        {
            if (!(dstBeing = beingManager->findBeing(msg.readLong())))
            {
                break;
            }

            int type = msg.readByte();
            int id = msg.readByte();

            switch (type) {
                case 1:
                    dstBeing->setHairStyle(id);
                    break;
                case 2:
                    dstBeing->setWeapon(id);
                    break;
                case 3:     // Change lower headgear for eAthena, pants for us
                    dstBeing->setVisibleEquipment(
                            Being::BOTTOMCLOTHES_SPRITE,
                            id);
                    break;
                case 4:     // Change upper headgear for eAthena, hat for us
                    dstBeing->setVisibleEquipment(
                            Being::HAT_SPRITE,
                            id);
                    break;
                case 5:     // Change middle headgear for eathena, armor for us
                     dstBeing->setVisibleEquipment(
                            Being::TOPCLOTHES_SPRITE,
                            id);
                    break;
                case 6:
                    dstBeing->setHairColor(id);
                    break;
                default:
                    logger->log("c3: %i\n", id); // unsupported
                    break;
            }
        }
            break;

        case SMSG_BEING_NAME_RESPONSE:
            if ((dstBeing = beingManager->findBeing(msg.readLong())))
            {
                dstBeing->setName(msg.readString(24));
            }
            break;

        case SMSG_PLAYER_UPDATE_1:
        case SMSG_PLAYER_UPDATE_2:
        case SMSG_PLAYER_MOVE:
            // An update about a player, potentially including movement.
            id = msg.readLong();
            speed = msg.readShort();
            msg.readShort();  // option 1
            msg.readShort();  // option 2
            msg.readShort();  // option
            job = msg.readShort();

            dstBeing = beingManager->findBeing(id);

            if (dstBeing == NULL)
            {
                dstBeing = beingManager->createBeing(id, job);
            }

            dstBeing->setWalkSpeed(speed);
            dstBeing->mJob = job;
            dstBeing->setHairStyle(msg.readShort());
            dstBeing->setWeaponById(msg.readShort());  // item id 1
            msg.readShort();  // item id 2
            headBottom = msg.readShort();

            if (msg.getId() == SMSG_PLAYER_MOVE)
            {
                msg.readLong(); // server tick
            }

            headTop = msg.readShort();
            headMid = msg.readShort();
            dstBeing->setHairColor(msg.readShort());
            msg.readShort();  // unknown
            msg.readShort();  // head dir
            msg.readLong();  // guild
            msg.readLong();  // emblem
            msg.readShort();  // manner
            msg.readByte();   // karma
            dstBeing->setSex(1 - msg.readByte());   // sex
            dstBeing->setVisibleEquipment(
                    Being::BOTTOMCLOTHES_SPRITE, headBottom);
            dstBeing->setVisibleEquipment(Being::HAT_SPRITE, headTop);
            dstBeing->setVisibleEquipment(Being::TOPCLOTHES_SPRITE, headMid);

            if (msg.getId() == SMSG_PLAYER_MOVE)
            {
                //Uint16 srcX, srcY, dstX, dstY;
                //msg.readCoordinatePair(srcX, srcY, dstX, dstY);
                //dstBeing->mX = srcX;
                //dstBeing->mY = srcY;
                //dstBeing->setDestination(dstX, dstY);
            }
            else
            {
                //Uint8 dir;
                //msg->readCoordinates(dstBeing->mX, dstBeing->mY, dir);
                //dstBeing->setDirection(dir);
            }

            msg.readByte();   // unknown
            msg.readByte();   // unknown

            if (msg.getId() == SMSG_PLAYER_UPDATE_1)
            {
                if (msg.readByte() == 2)
                {
                    dstBeing->setAction(Being::SIT);
                }
            }
            else if (msg.getId() == SMSG_PLAYER_MOVE)
            {
                msg.readByte(); // unknown
            }

            msg.readByte();   // Lv
            msg.readByte();   // unknown

            dstBeing->mWalkTime = tick_time;
            dstBeing->mFrame = 0;
            break;

        case 0x0119:
            // Change in players look
            logger->log("0x0119 %li %i %i %x %i\n", msg.readLong(),
                   msg.readShort(), msg.readShort(), msg.readShort(),
                   msg.readByte());
            break;
        */
    }
}

static void handleLooks(Being *being, MessageIn &msg)
{
    // Order of sent slots. Has to be in sync with the server code.
    static int const nb_slots = 4;
    static int const slots[nb_slots] =
        { Being::WEAPON_SPRITE, Being::HAT_SPRITE, Being::TOPCLOTHES_SPRITE,
          Being::BOTTOMCLOTHES_SPRITE };

    int mask = msg.readByte();
    if (mask & (1 << 8))
    {
        // The equipment has to be cleared first.
        being->setWeaponById(0);
        for (int i = 0; i < nb_slots; ++i)
        {
            if (slots[i] != Being::WEAPON_SPRITE)
                being->setVisibleEquipment(slots[i], 0);
        }
    }

    // Fill slots enumerated by the bitmask.
    for (int i = 0; i < nb_slots; ++i)
    {
        if (!(mask & (1 << i))) continue;
        int id = msg.readShort();
        if (slots[i] != Being::WEAPON_SPRITE)
            being->setVisibleEquipment(slots[i], id);
        else
            being->setWeaponById(id);
    }
}

void
BeingHandler::handleBeingEnterMessage(MessageIn &msg)
{
    int type = msg.readByte(); // type
    int id = msg.readShort();
    Being::Action action = (Being::Action)msg.readByte();
    Uint16 px = msg.readShort();
    Uint16 py = msg.readShort();

    switch (type)
    {
        case OBJECT_PLAYER:
        {
            std::string name = msg.readString();
            Being *being;
            if (player_node->getName() == name)
            {
                being = player_node;
                being->setId(id);
            }
            else
            {
                being = beingManager->createBeing(id, 0);
                being->setName(name);
            }
            being->setHairStyle(msg.readByte());
            being->setHairColor(msg.readByte());
            being->setSex(msg.readByte());
            being->mX = px;
            being->mY = py;
            being->setDestination(px, py);
            being->setAction(action);
            handleLooks(being, msg);
        } break;

        case OBJECT_MONSTER:
        {
            int monsterId = msg.readShort();
            Being *being;
            being = beingManager->createBeing(id, 1002 + monsterId);
            being->setWalkSpeed(150); // TODO
            being->mX = px;
            being->mY = py;
            being->setDestination(px, py);
            being->setAction(action);
        } break;
    }
}

void BeingHandler::handleBeingLeaveMessage(MessageIn &msg)
{
    Being *being = beingManager->findBeing(msg.readShort());
    if (!being) return;

    beingManager->destroyBeing(being);
}

void BeingHandler::handleBeingsMoveMessage(MessageIn &msg)
{
    while (msg.getUnreadLength())
    {
        Uint16 id = msg.readShort();
        Uint8 flags = msg.readByte();
        Being *being = beingManager->findBeing(id);
        int sx = 0, sy = 0, dx = 0, dy = 0;
        if (flags & MOVING_POSITION)
        {
            Uint16 sx2, sy2;
            msg.readCoordinates(sx2, sy2);
            sx = sx2 * 32 + 16;
            sy = sy2 * 32 + 16;
        }
        if (flags & MOVING_DESTINATION)
        {
            dx = msg.readShort();
            dy = msg.readShort();
            if (!(flags & MOVING_POSITION))
            {
                sx = dx;
                sy = dy;
            }
        }
        if (!being || !(flags & (MOVING_POSITION | MOVING_DESTINATION)))
        {
            continue;
        }
        if (abs(being->mX - sx) + abs(being->mY - sy) > 4 * 32)
        {
            // Too large a desynchronization.
            being->mX = sx;
            being->mY = sy;
            being->setDestination(dx, dy);
        }
        else if (!(flags & MOVING_POSITION))
        {
            being->setDestination(dx, dy);
        }
        else if (!(flags & MOVING_DESTINATION))
        {
            being->adjustCourse(sx, sy);
        }
        else
        {
            being->adjustCourse(sx, sy, dx, dy);
        }
    }
}

void BeingHandler::handleBeingAttackMessage(MessageIn &msg)
{
    Being *being = beingManager->findBeing(msg.readShort());
    if (!being) return;

    being->setAction(Being::ATTACK);
    being->setDirection(msg.readByte());
}

void BeingHandler::handleBeingsDamageMessage(MessageIn &msg)
{
    while (msg.getUnreadLength())
    {
        Being *being = beingManager->findBeing(msg.readShort());
        int damage = msg.readShort();
        if (being)
        {
            being->takeDamage(damage);
        }
    }
}

void BeingHandler::handleBeingActionChangeMessage(MessageIn &msg)
{
    Being* being = beingManager->findBeing(msg.readShort());
    if (!being) return;

    being->setAction((Being::Action) msg.readByte());
}

void BeingHandler::handleBeingLooksChangeMessage(MessageIn &msg)
{
    Being *being = beingManager->findBeing(msg.readShort());
    if (!being) return;
    handleLooks(being, msg);
}