青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

天行健 君子當自強而不息

Putting Together a Full Game(14)

 

Controlling Characters

The characters are the heart and soul of your game.

You derive the character controller in order to
control the player of the game and to collision-check a character's movements
against the maps. For The Tower, you can use a derived character controller,
to manage all your game's characters. The first step to
using the character controller in a game is to derive your own class from cCharController:

class cGameChars : public cCharController
{
private:
    cApp*   m_app;

    
////////////////////////////////////////////////////////////////////////////////

public:
    
void attach_app(cApp* app)
    {
        m_app = app;
    }

private:
    
virtual void drop_money(float x_pos, float y_pos, float z_pos, long quantity);
    
virtual bool gain_exp(sCharacter* character, long amount);
    
virtual bool pc_teleport(sCharacter* character, const sSpell* spell);

    
virtual void pc_update(sCharacter* character, long elapsed,
                           
float* x_move, float* y_move, float* z_move);

    
virtual bool validate_move(sCharacter* character, float* x_move, float* y_move, float* z_move);
    
virtual void play_action_sound(const sCharacter* character);
};

The cGameChars class comes with only one public function, attach_app. You use the cGameChars
function to set the application class pointer in the cGameChars class instance. In addition,
the cGameChars class overrides only the functions used to move the player and to validate
all character movements. The remaining functions come into play when the player
gains experience points from combat or teleports the character with a spell, when
the character controller plays a sound, when a monster drops some money, or
when a monster drops an item after being killed.

Because the derived character controller class requires access to the application
class, you must precede all calls to the cGameChars class with a call to the attach_app function.
The attach_app function takes one argument—the pointer to the application class.

The other functions, such as gain_exp, drop_money, tell the game engine
that a monster was killed and that the game needs to reward the player with experience,
money from a dying monster. These rewards are pushed
aside until combat ends, at which point, the application class's end_of_combat function
processes them.

void cGameChars::drop_money(float x_pos, float y_pos, float z_pos, long quantity)
{
    m_app->m_combat_money += quantity;
}

bool cGameChars::gain_exp(sCharacter* character, long amount)
{
    m_app->m_combat_exp += amount;

    
return false;   // do not display message
}

bool cGameChars::pc_teleport(sCharacter* character, const sSpell* spell)
{
    
// teleport player to town
    m_app->teleport_player(1, 100.0f, 0.0f, -170.0f);

    
return true;
}

///////////////////////////////////////////////////////////////////////////////////////////////////

bool cGameChars::validate_move(sCharacter* character, float* x_move, float* y_move, float* z_move)
{
    
float pos_x = character->pos_x;
    
float pos_y = character->pos_y;
    
float pos_z = character->pos_z;

    
// check against terrain mesh for collision
    if(m_app->check_intersect(pos_x, pos_y + 8.0f, pos_z,
                              pos_x + *x_move, pos_y + 8.0f + *y_move, pos_z + *z_move,
                              NULL))
        
return false;

    
// get new height
    float height = m_app->get_height_below(pos_x + *x_move, pos_y + 32.0f, pos_z + *z_move);
    *y_move = height - pos_y;

    
// check against barriers and clear movement if any
    if(m_app->m_barrier.get_barrier(pos_x + *x_move, pos_y + *y_move, pos_z + *z_move))
        (*x_move) = (*y_move) = (*z_move) = 0.0f;

    
return true;
}

///////////////////////////////////////////////////////////////////////////////////////////////////

void cGameChars::play_action_sound(const sCharacter* character)
{
    
long sound = 0;

    
// play sound based on character type and action
    switch(character->action)
    {
    
case CHAR_ATTACK:
        m_app->play_sound(character->id == ID_PLAYER ? SOUND_CHAR_ATTACK : SOUND_MONSTER_ATTACK);
        
break;

    
case CHAR_SPELL:
        
if(character->spell_index == SPELL_FIREBALL)
            sound = SOUND_FIREBALL;
        
else if(character->spell_index == SPELL_ICE)
            sound = SOUND_ICE;
        
else if(character->spell_index == SPELL_HEAL)
            sound = SOUND_HEAL;
        
else if(character->spell_index == SPELL_TELEPORT)
            sound = SOUND_TELEPORT;
        
else if(character->spell_index == SPELL_GROUNDBALL)
            sound = SOUND_GROUNDBALL;
        
else if(character->spell_index == SPELL_CONCUSSION)
            sound = SOUND_CONCUSSION;
        
else if(character->spell_index == SPELL_EVIL_FORCE)
            sound = SOUND_EVIL_FORCE;

        m_app->play_sound(sound);
        
break;

    
case CHAR_HURT:
        m_app->play_sound(character->id == ID_PLAYER ? SOUND_CHAR_HURT : SOUND_MONSTER_HURT);
        
break;

    
case CHAR_DIE:
        m_app->play_sound(character->id == ID_PLAYER ? SOUND_CHAR_DIE : SOUND_MONSTER_DIE);
        
break;
    }
}

The pc_update is the main function of interest here. It determines which keys the
player is pressing and what mouse button is being pressed. Now, take this function
apart to see what makes it tick:


void cGameChars::pc_update(sCharacter* character, long elapsed,
                           
float* x_move, float* y_move, float* z_move)
{
    
if(elapsed == 0)    // do not update if no elapsed time
        return;

    
// rotate character

    
if(m_app->m_keyboard.get_key_state(KEY_LEFT) || m_app->m_keyboard.get_key_state(KEY_A))
    {
        character->direction -= elapsed / 1000.0f * 4.0f;
        character->action = CHAR_MOVE;
    }

    
if(m_app->m_keyboard.get_key_state(KEY_RIGHT) || m_app->m_keyboard.get_key_state(KEY_D))
    {
        character->direction += elapsed / 1000.0f * 4.0f;
        character->action = CHAR_MOVE;
    }

    
// calculate move length along x,z axis

    
if(m_app->m_keyboard.get_key_state(KEY_UP) || m_app->m_keyboard.get_key_state(KEY_W))
    {
        
float speed = elapsed / 1000.0f * m_app->m_game_chars.get_speed(character);

        *x_move = sin(character->direction) * speed;
        *z_move = cos(character->direction) * speed;

        character->action = CHAR_MOVE;
    }

    
// process attack or talk action
    if(m_app->m_mouse.get_button_state(MOUSE_LBUTTON))
    {
        
// see which character is being pointed at and make sure if it is a monster.

        sCharacter* target = m_app->get_char_at(m_app->m_mouse.get_x_pos(), m_app->m_mouse.get_y_pos());

        
if(target != NULL)
        {
            
if(target->type == CHAR_NPC)    // handle talking to npcs
            {
                
// no distance checks, just process their script.

                
char filename[MAX_PATH];
                sprintf(filename, "..\\Data\\Char%lu.mls", target->id);

                m_app->m_game_script.execute(filename);
                
return// do not process anymore
            }
            
else if(target->type == CHAR_MONSTER)   // handle attacking monsters
            {
                
// get distance to target
                float x_diff = fabs(target->pos_x - character->pos_x);
                
float y_diff = fabs(target->pos_y - character->pos_y);
                
float z_diff = fabs(target->pos_z - character->pos_z);

                
float dist = x_diff * x_diff + y_diff * y_diff + z_diff * z_diff;

                
// offset dist by target's radius
                float monster_radius = get_xz_radius(target);
                dist -= (monster_radius * monster_radius);

                
float attack_range = get_xz_radius(character) + character->char_def.attack_range;

                
// only perform attack if target in range
                if(dist <= (attack_range * attack_range))
                {
                    target->attacker  = character;
                    character->victim = target;

                    
// face victim
                    x_diff = target->pos_x - character->pos_x;
                    z_diff = target->pos_z - character->pos_z;
                    character->direction = atan2(x_diff, z_diff);

                    m_app->m_game_chars.set_char_action(character, CHAR_ATTACK, 0);
                }
            }
        }
    }

    
long spell_index = -1;

    
// cast magic spll based on NUM key pressed

    
if(m_app->m_keyboard.get_key_state(KEY_1))
    {
        m_app->m_keyboard.m_locks[KEY_1] = 
true;
        spell_index = SPELL_FIREBALL;
    }
    
else if(m_app->m_keyboard.get_key_state(KEY_2))
    {
        m_app->m_keyboard.m_locks[KEY_2] = 
true;
        spell_index = SPELL_ICE;
    }
    
else if(m_app->m_keyboard.get_key_state(KEY_3))
    {
        m_app->m_keyboard.m_locks[KEY_3] = 
true;
        spell_index = SPELL_HEAL;
    }
    
else if(m_app->m_keyboard.get_key_state(KEY_4))
    {
        m_app->m_keyboard.m_locks[KEY_4] = 
true;
        spell_index = SPELL_TELEPORT;
    }
    
else if(m_app->m_keyboard.get_key_state(KEY_5))
    {
        m_app->m_keyboard.m_locks[KEY_5] = 
true;
        spell_index = SPELL_GROUNDBALL;
    }

    
// cast spell if commanded
    if(spell_index != -1)
    {
        sSpell* spell = m_app->m_game_spells.get_spell(spell_index);

        
// only cast if spell known and has enough mana points
        if(char_can_spell(character, spell, spell_index))
        {
            
// see which character is being pointed
            sCharacter* target = m_app->get_char_at(m_app->m_mouse.get_x_pos(), m_app->m_mouse.get_y_pos());

            
if(target && target->type == CHAR_MONSTER)
            {
                
// get distance to target

                
float x_diff = fabs(target->pos_x - character->pos_x);
                
float y_diff = fabs(target->pos_y - character->pos_y);
                
float z_diff = fabs(target->pos_z - character->pos_z);

                
float dist = x_diff * x_diff + y_diff * y_diff + z_diff * z_diff;

                
// offset dist by target's radius
                float target_radius = get_xz_radius(target);
                dist -= (target_radius * target_radius);

                
float spell_range = get_xz_radius(character) + spell->max_dist;
        
                
// only perform spell if target in range
                if(dist <= (spell_range * spell_range))
                {
                    character->spell_index = spell_index;
                    character->target_type = CHAR_MONSTER;
                    character->target_x    = target->pos_x;
                    character->target_y    = target->pos_y;
                    character->target_z    = target->pos_z;

                    *x_move = *y_move = *z_move = 0.0f;     
// clear movement

                    // face victim
                    x_diff = target->pos_x - character->pos_x;
                    z_diff = target->pos_z - character->pos_z;

                    character->direction = atan2(x_diff, z_diff);

                    set_char_action(character, CHAR_SPELL, 0);
                }
            }
            
else if(target == character)
            {
                
if((spell_index == SPELL_HEAL && character->health_points < character->char_def.health_points) ||
                   (spell_index == SPELL_TELEPORT))
                {
                    character->spell_index = spell_index;
                    character->target_type = CHAR_PC;
                    character->target_x    = target->pos_x;
                    character->target_y    = target->pos_y;
                    character->target_z    = target->pos_z;

                    *x_move = *y_move = *z_move = 0.0f;     
// clear movement

                    set_char_action(character, CHAR_SPELL, 0);
                }
            }
        }
    }

    
// enter status frame if right mouse buttom pressed
    if(m_app->m_mouse.get_button_state(MOUSE_RBUTTON))
    {
        m_app->m_mouse.m_locks[MOUSE_RBUTTON] = 
true;
        m_app->m_state_manager.push(status_frame, m_app);
    }
}

pc_update starts by first determining whether an update is in order (based on
whether any time has elapsed) and continues by determining which keys (if any)
are pressed on the keyboard. If the up arrow key is pressed, the character moves
forward, whereas if the left or right arrow keys are pressed, the character’s direction
is modified.

For each movement that the player performs, such as walking forward or turning
left and right, you need to assign the CHAR_MOVE action to the player's character.
Notice that even though pressing left or right immediately rotates the player’s character,
the code does not immediately modify the character’s coordinates. Instead,
you store the direction of travel in the x_move and z_move variables.

You then determine whether the player has clicked the left mouse button. Remember
from the design of the sample game that clicking the left mouse button on a nearby
character either attacks the character (if the character is a monster) or speaks to the
character (if the character is an NPC).

The portion of code just shown calls upon the get_char_at function, which scans
for the character that is positioned under the mouse cursor. If a character is found,
you determine which type of character it is; if it is an NPC, you execute the appropriate
character's script.

On the other hand, if the character clicked is a monster and that monster is within
attack range, you initiate an attack action.

Coming up to the end of the pc_update function, the controller needs to determine
whether a spell has been cast at a nearby character. In the game, positioning the
mouse cursor over a character and pressing one of the number keys (from 1 through
5) casts a spell.

If a spell was cast, the controller determines whether the player knows the spell and
has enough mana to cast the spell and whether the target character is in range.

At this point, the controller has determined that the spell can be cast. You need
to store the coordinates of the target, the number of the spell being cast, and the
player's action in the structure pointed to by the Character pointer.

To finish up the player character update, the controller determines whether the
player clicked the right mouse button, which opens the character’s status window
(by pushing the character-status state onto the state stack).

posted on 2007-12-30 00:24 lovedday 閱讀(333) 評論(0)  編輯 收藏 引用


只有注冊用戶登錄后才能發表評論。
網站導航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


公告

導航

統計

常用鏈接

隨筆分類(178)

3D游戲編程相關鏈接

搜索

最新評論

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            欧美成人资源| 美女精品自拍一二三四| 日韩亚洲国产精品| 亚洲精品系列| 亚洲在线观看免费| 欧美日韩黄色大片| 久久九九久精品国产免费直播| 亚洲欧美韩国| 欧美精品粉嫩高潮一区二区| 蜜臀av一级做a爰片久久| 久久综合网hezyo| 国产午夜精品一区二区三区欧美| 亚洲精选国产| 欧美成人一品| 欧美激情一区二区三区在线视频观看| 一区二区三区欧美日韩| 欧美在线观看一区| 韩日在线一区| 一区二区精品在线| 亚洲色图制服丝袜| 欧美一进一出视频| 国产精品久久午夜| 国外成人免费视频| 亚洲欧美日韩在线| 精品va天堂亚洲国产| 中国亚洲黄色| 免费亚洲一区| 亚洲欧美国产一区二区三区| 欧美乱妇高清无乱码| 亚洲国产成人久久| 久久综合一区| 99视频在线精品国自产拍免费观看| 欧美www视频在线观看| 亚洲国产日韩欧美在线99| 亚洲午夜精品福利| 亚洲欧美视频在线观看视频| 欧美日韩免费网站| 亚洲视频在线观看一区| 欧美成人按摩| 欧美高清视频在线| 亚洲人成人一区二区三区| 久久国产免费| 欧美一区二区精品久久911| 欧美日韩另类国产亚洲欧美一级| 小处雏高清一区二区三区| 午夜精品在线| 激情欧美国产欧美| 蜜桃av综合| 免费在线看成人av| 亚洲视频1区| 久久久久久久一区二区| 亚洲三级免费观看| 欧美在线视频一区二区| 欧美在线视频在线播放完整版免费观看 | 亚洲视频在线观看三级| 欧美一区=区| 在线日韩欧美视频| 91久久久久久久久久久久久| 免费日韩成人| 久久久久久久一区二区| 欧美午夜无遮挡| 亚洲成色最大综合在线| 国产精品国产三级国产aⅴ9色| 欧美1区2区| 国产精品视频免费观看| 亚洲女女女同性video| 免费久久99精品国产| 亚洲三级免费电影| 免费久久久一本精品久久区| 亚洲一区二区少妇| 欧美精品手机在线| 久久精品国产v日韩v亚洲| 久久色在线观看| 精品动漫3d一区二区三区| 欧美电影打屁股sp| 国产欧美日韩亚州综合| 亚洲素人在线| 亚洲欧洲精品一区二区三区不卡| 国产亚洲欧美另类一区二区三区| 久久久噜噜噜久久久| 在线性视频日韩欧美| 亚洲男人av电影| 欧美欧美天天天天操| 欧美成人dvd在线视频| 亚欧美中日韩视频| 免费欧美日韩国产三级电影| 欧美在线精品免播放器视频| 免费欧美在线| 久久夜色精品国产欧美乱极品| 国产欧美欧美| 亚洲综合日韩中文字幕v在线| 欧美激情综合五月色丁香| 久久国产精品久久久久久电车| 在线中文字幕不卡| 午夜精品久久久久久久白皮肤 | 欧美激情亚洲自拍| 亚洲精品欧美一区二区三区| 99热精品在线| 欧美视频日韩| 日韩一级在线观看| 国内精品国产成人| 久久久久综合一区二区三区| 久久久国产精品亚洲一区| 免费观看成人| 日韩视频免费在线| 亚洲免费中文字幕| 亚洲欧美精品一区| 好看的日韩视频| 欧美大胆成人| 久久国产精品高清| 影音先锋在线一区| 欧美激情麻豆| 一区二区日韩免费看| 狠狠狠色丁香婷婷综合久久五月| 在线观看视频日韩| 亚洲欧洲中文日韩久久av乱码| 亚洲精品一区二区网址| 亚洲免费视频网站| 嫩草国产精品入口| 亚洲免费观看| 国产日韩欧美二区| 欧美激情日韩| 午夜精品一区二区三区在线视 | 国产亚洲欧洲997久久综合| 欧美成人久久| 宅男噜噜噜66一区二区66| 久久精品首页| 一区二区久久| 亚洲欧美日韩精品一区二区 | 日韩西西人体444www| 欧美一区国产在线| 久热精品视频在线免费观看 | 久久精品1区| 亚洲欧美在线视频观看| 女生裸体视频一区二区三区| 在线视频精品一区| 欧美精品一区二| 永久免费视频成人| 欧美亚洲一区二区在线| 亚洲高清网站| 亚洲欧美日韩国产一区二区三区| 麻豆精品91| 欧美成人首页| 久久精品在线观看| 国产精品萝li| 亚洲欧洲在线看| 久久人人看视频| 亚洲精品美女在线观看| 欧美自拍偷拍午夜视频| 亚洲女人天堂成人av在线| 亚洲激情一区二区| 狠狠久久婷婷| 国产专区欧美专区| 亚洲一区国产精品| 日韩特黄影片| 亚洲人成在线免费观看| 久久免费黄色| 伊人久久综合97精品| 久久久7777| 久久久999精品视频| 国产综合久久| 黄色在线一区| 国外成人在线视频网站| 韩日午夜在线资源一区二区| 国产精品综合| 午夜视频一区| 久久露脸国产精品| 欧美专区18| 美日韩精品免费| 欧美成人午夜| 亚洲欧洲综合另类| 99re6这里只有精品视频在线观看| 99亚洲一区二区| 免费在线观看日韩欧美| 欧美日韩国产免费观看| 99国产精品99久久久久久粉嫩| 欧美电影在线观看完整版| 欧美高清在线| 欧美精品高清视频| 在线一区二区三区四区五区| 国产精品夜色7777狼人| 黄色一区二区在线| 亚洲精品一级| 午夜精品福利一区二区蜜股av| 久久成人资源| 美女999久久久精品视频| 亚洲高清一区二区三区| 亚洲国产精品国自产拍av秋霞 | 亚洲综合国产激情另类一区| 欧美一区二区日韩| 久久大综合网| 亚洲国产成人精品视频| 欧美日韩一二三四五区| 9l国产精品久久久久麻豆| 欧美日韩综合精品| 好吊色欧美一区二区三区四区| 欧美精品福利在线| 亚洲一区二区日本| 免费观看亚洲视频大全| 亚洲——在线|