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

天行健 君子當自強而不息

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| 亚洲国产精品一区在线观看不卡| 欧美亚一区二区| 欧美日韩精品在线| 欧美日本中文| 欧美噜噜久久久xxx| 欧美激情一区二区三区高清视频| 蜜臀av在线播放一区二区三区| 美国十次了思思久久精品导航| 鲁大师影院一区二区三区| 亚洲精品久久久久久下一站| 亚洲精品在线电影| 亚洲精品人人| 午夜精品成人在线| 久久免费黄色| 欧美日韩黄视频| 国产精品一区二区在线观看网站 | 久久野战av| 欧美电影免费观看| 夜夜夜久久久| 久久久精品欧美丰满| 欧美精品三级在线观看| 国产精品一级| 亚洲精品乱码久久久久久黑人| 亚洲伊人网站| 欧美mv日韩mv国产网站| 在线一区二区三区做爰视频网站| 久久成人一区| 欧美色123| 亚洲高清久久| 午夜在线精品偷拍| 亚洲成人资源| 日韩午夜视频在线观看| 久久久久久久久久久一区| 欧美日韩国产综合网| 国产亚洲免费的视频看| av72成人在线| 欧美黄色片免费观看| 亚洲一区二区免费在线| 麻豆免费精品视频| 国产一区二区精品丝袜| 亚洲一区二区网站| 欧美日韩一区二区精品| 亚洲国产第一| 久久电影一区| 欧美黑人在线播放| 蜜臀av国产精品久久久久| 好吊日精品视频| 国产精品99久久久久久宅男 | 欧美激情精品久久久久| 男男成人高潮片免费网站| 午夜精品偷拍| 91久久精品www人人做人人爽| 久久婷婷蜜乳一本欲蜜臀| 亚洲国产日韩在线一区模特| 国产欧美日本在线| 国产噜噜噜噜噜久久久久久久久| 久久综合九色| 国产精品av免费在线观看| 国产精品三上| 尤物精品在线| 性欧美videos另类喷潮| 亚洲福利精品| 亚洲一区黄色| 一区二区免费在线视频| 欧美高清在线视频| 亚洲国产视频a| 欧美成人精品激情在线观看| 欧美在线高清视频| 午夜久久久久久| 国产欧美精品一区二区三区介绍| 亚洲天堂成人| 亚洲一级高清| 欧美天天综合网| 欧美一区网站| 久久精品国语| 亚洲精品免费一二三区| 亚洲精品综合久久中文字幕| 欧美日韩999| 亚洲午夜视频在线观看| 欧美制服丝袜第一页| 国产精品美女久久久久av超清| 亚洲国产精品123| 欧美激情视频一区二区三区不卡| 精品成人一区二区| 在线视频一区二区| 亚洲欧美日韩精品| 欧美日韩国产成人在线免费| 久久成人精品电影| 国产女主播一区二区三区| 亚洲理论在线观看| 亚洲一区制服诱惑| 国产伦精品一区二区三区在线观看 | 亚洲女爱视频在线| 中日韩高清电影网| 国产午夜久久久久| 亚洲国产日韩欧美| 亚洲人体偷拍| 欧美网站在线观看| 午夜在线精品偷拍| 另类av一区二区| 午夜精品视频一区| 久久亚洲精品伦理| 亚洲综合色激情五月| 久久视频精品在线| 午夜视频一区在线观看| 久热这里只精品99re8久| 亚洲线精品一区二区三区八戒| 欧美在线视频免费| 这里只有精品视频| 久久久久久久久久久久久久一区| 中文精品99久久国产香蕉| 欧美一区深夜视频| 亚洲免费伊人电影在线观看av| 久久人人97超碰国产公开结果| 亚洲欧美不卡| 欧美精品成人一区二区在线观看 | 亚洲欧美日韩精品久久| 国产精品电影观看| 国产日韩欧美一区二区三区在线观看 | 亚洲人成网站在线播| 午夜精品视频网站| 一区二区三区精密机械公司| 久久亚洲春色中文字幕| 欧美在线视频免费| 国产精品剧情在线亚洲| 亚洲精品视频二区| 亚洲区国产区| 两个人的视频www国产精品| 久久久久国产精品厨房| 国产精品亚洲产品| 亚洲网站在线| 亚洲欧美日韩精品在线| 欧美日韩在线播放一区二区| 亚洲国产成人精品久久久国产成人一区 | 亚洲韩国一区二区三区| 欧美在线一级视频| 午夜在线精品| 国产亚洲一区二区三区| 午夜精品国产| 久久精品在线免费观看| 国产精品ⅴa在线观看h| 亚洲免费不卡| 一区二区三区成人| 欧美黑人在线观看| 91久久精品美女高潮| 狠狠色丁香婷婷综合影院| 久久成人综合视频| 欧美高清视频| 99伊人成综合| 欧美一级大片在线免费观看| 欧美国产在线视频| 91久久线看在观草草青青| 欧美另类在线播放| 亚洲视频精品| 久久美女艺术照精彩视频福利播放| 国产伦精品一区二区三区四区免费| 亚洲欧美国产制服动漫| 国产精品婷婷| 久久精品91久久久久久再现| 欧美激情第五页| 亚洲欧美欧美一区二区三区| 麻豆久久久9性大片| 一本不卡影院| 91久久夜色精品国产九色| 欧美日韩另类视频| 免费观看久久久4p| 亚洲另类一区二区| 99视频一区二区三区| 美女精品在线| 91久久精品久久国产性色也91| 日韩一本二本av| 狠狠色噜噜狠狠色综合久| 亚洲精品在线观看免费| 国产精品看片资源| 午夜精品久久久久久99热软件| 亚洲一区中文字幕在线观看| 国产欧美91| 亚洲人成人一区二区在线观看| 国产精品婷婷午夜在线观看| 亚洲国产精品一区在线观看不卡 | 国产精品第2页| 在线观看国产成人av片| 午夜精品久久久久久久| 欧美国产国产综合| 久久精品国产亚洲a| 一区二区毛片| 亚洲一区二区三区激情| 99re6热只有精品免费观看 | 在线视频欧美精品| 亚洲欧美激情视频| 欧美喷潮久久久xxxxx| 亚洲一级片在线观看| 亚洲国产欧美一区二区三区同亚洲| 欧美主播一区二区三区美女 久久精品人 | 亚洲黄色影片| 在线成人免费视频| 久久久久久久久久看片| 久久综合给合|