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

天行健 君子當自強而不息

Controlling Players and Characters(36)

Notice that I keep talking about removing characters from the list. What about all
the hard work you’ve put into your PCs—how are you ever going to store their
achievements for later loading? With the following set of saving and loading functions,
of course!

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

bool cCharController::save_char(long id, const char* filename)
{
    sCharacter* character = get_char(id);
    
if(character == NULL)
        
return false;

    FILE* fp = fopen(filename, "wb");
    
if(fp == NULL)
        
return false;

    
// output character data
    fwrite(&character->char_def,      1, sizeof(character->char_def),      fp);
    fwrite(&character->health_points, 1, 
sizeof(character->health_points), fp);
    fwrite(&character->mana_points,   1, 
sizeof(character->mana_points),   fp);
    fwrite(&character->ailments,      1, 
sizeof(character->ailments),      fp);
    fwrite(&character->pos_x,         1, 
sizeof(character->pos_x),         fp);
    fwrite(&character->pos_y,         1, 
sizeof(character->pos_y),         fp);
    fwrite(&character->pos_z,         1, 
sizeof(character->pos_z),         fp);
    fwrite(&character->direction,     1, 
sizeof(character->direction),     fp);

    fclose(fp);

    
// save inventory
    if(character->char_ics)
    {
        
char ics_file[MAX_PATH];
        sprintf(ics_file, "ICS%s", filename);
        character->char_ics->save(ics_file);
    }

    
return true;
}

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

bool cCharController::load_char(long id, const char* filename)
{
    sCharacter* character = get_char(id);
    
if(character == NULL)
        
return false;

    FILE* fp = fopen(filename, "rb");
    
if(fp == NULL)
        
return false;

    
// read in character data
    fread(&character->char_def,      1, sizeof(character->char_def),      fp);
    fread(&character->health_points, 1, 
sizeof(character->health_points), fp);
    fread(&character->mana_points,   1, 
sizeof(character->mana_points),   fp);
    fread(&character->ailments,      1, 
sizeof(character->ailments),      fp);
    fread(&character->pos_x,         1, 
sizeof(character->pos_x),         fp);
    fread(&character->pos_y,         1, 
sizeof(character->pos_y),         fp);
    fread(&character->pos_z,         1, 
sizeof(character->pos_z),         fp);
    fread(&character->direction,     1, 
sizeof(character->direction),     fp);

    fclose(fp);

    
// load inventory
    if(character->char_ics)
    {
        
char ics_file[MAX_PATH];
        sprintf(ics_file, "ICS%s", filename);
        character->char_ics->load(ics_file);
    }

    
return true;
}

Both of the two preceding functions take the character’s identification number to
save or load, as well as the filename to use.

That about rounds up the functions used to prepare, add, and remove the characters
from the game. Now it’s time to get them all moving around performing their actions.
Previously you saw the functions used to update the individual character types; now
comes the single function you’ll call to update all characters at once:

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

void cCharController::update(long elapsed)
{    
    
if(m_root_char == NULL)
        
return;

    
static long effect_counter = 0;
    effect_counter += elapsed;

    
float x_move, y_move, z_move;
    sCharacter* next_char;

    
// loop through all characters
    for(sCharacter* char_ptr = m_root_char; char_ptr != NULL; char_ptr = next_char)
    {
        next_char = char_ptr->next;     
// remember next character

        // only update if enabled, not asleep or paralyzed.
        if(! char_ptr->update_enable)
            
continue;

        
// update action timer if in use
        if(char_ptr->action_timer != 0)
        {
            char_ptr->action_timer -= elapsed;

            
if(char_ptr->action_timer < 0)
                char_ptr->action_timer = 0;
        }

        
if(char_ptr->msg_timer > 0)
            char_ptr->msg_timer -= elapsed;

        
// reset charge counter if attacking, spell, or item.
        if(char_ptr->action == CHAR_ATTACK || char_ptr->action == CHAR_SPELL || char_ptr->action == CHAR_ITEM)
            char_ptr->charge = 0.0f;

        
// kill character if no health left
        if(char_ptr->health_points <= 0 && char_ptr->action != CHAR_DIE)
            set_char_action(char_ptr, CHAR_DIE, 0);

        
bool to_process = true;     // mark that processing can continue later on
        bool dead_char  = false;    // mark character as still alive

        // do not allow on update if asleep or paralyzed
        if((char_ptr->ailments & AILMENT_SLEEP) || (char_ptr->ailments & AILMENT_PARALYZE))
            to_process = 
false;

        
// process actions
        if(char_ptr->action_timer == 0)
        {
            
switch(char_ptr->action)
            {
            
case CHAR_ATTACK:
                
if(to_process)
                    attack(char_ptr, char_ptr->victim);

                
break;

            
case CHAR_SPELL:
                
if(to_process)
                    m_spell_controller->add(char_ptr);

                
break;

            
case CHAR_ITEM:
                
if(to_process)
                    use_item(char_ptr, char_ptr, char_ptr->item_index, char_ptr->char_item);

                
break;

            
case CHAR_DIE:
                death(char_ptr->attacker, char_ptr);
                dead_char  = 
true;
                to_process = 
false;
                
break;
            }
        }

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

        
// only continue if allowed (in case character died)
        if(to_process)
        {
            
// only allow updates if lock/timer not in use
            if(char_ptr->action_timer == 0 && !char_ptr->is_lock)
            {                    
                char_ptr->action = CHAR_IDLE;   
// reset action

                
if(char_ptr->type == CHAR_PC)
                    pc_update(char_ptr, elapsed, &x_move, &y_move, &z_move);
                
else
                    npc_monster_update(char_ptr, elapsed, &x_move, &y_move, &z_move);

                
// check for validity of movement (clear if invalid)
                if(!check_move(char_ptr, &x_move, &y_move, &z_move))
                {
                    x_move = y_move = z_move = 0.0f;
                    char_ptr->action = CHAR_IDLE;
                }
            }

            process_update(char_ptr, x_move, y_move, z_move);

            char_ptr->charge += (elapsed/1000.0f * get_charge_rate(char_ptr));

            
if(char_ptr->charge > 100.0f)
                char_ptr->charge = 100.0f;
        }

        
// process timed ailments (only in live characters)
        if(!dead_char && char_ptr->ailments)
        {
            
// sleeping characters have 4% to wake up
            if((char_ptr->ailments & AILMENT_SLEEP) && rand()%100 < 4)
                char_ptr->ailments &= ~AILMENT_SLEEP;

            
// paralyzed character have 2% chance to recover
            if((char_ptr->ailments & AILMENT_PARALYZE) && rand()%100 < 2)
                char_ptr->ailments &= ~AILMENT_PARALYZE;

            
// poison removes 2 hp every 4 seconds
            if((char_ptr->ailments & AILMENT_POISON) && effect_counter >= 4000)
            {
                char_ptr->health_points -= 2;
                set_char_msg(char_ptr, "Poison -2 HP", 500, D3DCOLOR_RGBA(0, 255, 64, 255));
            }
        }
    }

    
// reset effect counter (after 4 seconds)
    if(effect_counter >= 4000)
        effect_counter = 0;
}

The Update function is called once every frame. Taking a single argument (the time
elapsed since the last update), the Update function calls upon each character’s
respective update function, validates each character’s movements and actions, and
wraps up by processing the actions. Then a call to Render is in order to display all
characters visible within the specified frustum.
///////////////////////////////////////////////////////////////////////////////////////////////////

void cCharController::render(long elapsed, cFrustum* frustum, float z_dist)
{
    m_frustum = frustum;

    
// construct the viewing frustum (if none passed)
    if(m_frustum == NULL)
    {
        cFrustum view_frustum;
        view_frustum.create(z_dist);
        m_frustum = &view_frustum;
    }

    DWORD time = 0;

    
// get time to update animations (30fps) if elapsed value passed == -1
    if(elapsed == -1)
        time = timeGetTime() / 30;

    
// variables for printing messages
    bool got_matrices = false;
    D3DXMATRIX mat_world, mat_view, mat_proj;
    D3DVIEWPORT9 viewport;

    
// loop through each character and draw
    for(sCharacter* char_ptr = m_root_char; char_ptr != NULL; char_ptr = char_ptr->next)
    {
        
// update animation based on elapsed time passed
        if(elapsed != -1)
        {
            char_ptr->last_anim_time += elapsed/30;
            time = char_ptr->last_anim_time;
        }

        
float max_y, radius;
        cObject& 
object = char_ptr->object;
        
object.get_bounds(NULL, NULL, NULL, NULL, &max_y, NULL, &radius);

        
// draw character if in viewing frustum
        if(! m_frustum->is_sphere_in(object.get_x_pos(), object.get_y_pos(), object.get_z_pos(), radius))
            
continue;

        
object.update_anim(time, true);
        
object.render();
        
        
if(char_ptr->char_def.weapon != -1)
            char_ptr->weapon_object.render();

        
// draw message if needed
        if(char_ptr->msg_timer > 0)
        {
            
// get the matrices and viewport if not done already
            if(!got_matrices)
            {
                got_matrices = 
true;

                
// get the world, projection, and view transformations.
                D3DXMatrixIdentity(&mat_world);
                get_display_view_matrix(&mat_view);
                get_display_proj_matrix(&mat_proj);
                get_display_viewport(&viewport);
            }

            D3DXVECTOR3 pos;
            D3DXVECTOR3 src_pos(char_ptr->pos_x, char_ptr->pos_y + (max_y * 0.5f), char_ptr->pos_z);

            
// project a 3D vector from object space into screen space
            D3DXVec3Project(&pos, &src_pos, &viewport, &mat_proj, &mat_view, &mat_world);

            draw_font(m_font, char_ptr->msg, pos.x, pos.y, 0, 0, char_ptr->msg_color, DT_LEFT);
        }
    }
}

With render, you have a few optional arguments. You use the first one to control
the animation timing of the characters. In a task-switchable environment such as
Windows, merely using the time elapsed from the last processed frame is unacceptable;
you must instead specify a fixed amount of time passed and ensure that your
game engine sticks to updates at that rate.

As for the viewing frustum pointer, the application can provide its own pre-created
object, or pass NULL (and an optional Z-distance) to create its own frustum.

 

posted on 2007-12-04 18:55 lovedday 閱讀(299) 評論(0)  編輯 收藏 引用

公告

導航

統計

常用鏈接

隨筆分類(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>
            亚洲毛片一区二区| 亚洲一二三级电影| 久久这里只有精品视频首页| 国内激情久久| 男人的天堂成人在线| 男女激情久久| 免费亚洲网站| 99精品欧美一区| 亚洲一卡二卡三卡四卡五卡| 国产日韩专区| 亚洲第一福利在线观看| 亚洲图片在线| 国产一区二区日韩精品欧美精品| 尤物yw午夜国产精品视频明星| 一区二区三区四区国产精品| 亚洲视频在线观看免费| 美女网站久久| 亚洲调教视频在线观看| 午夜伦理片一区| 亚洲免费观看视频| 免费日韩成人| 欧美网站在线观看| 噜噜噜久久亚洲精品国产品小说| 一区二区三区偷拍| 黄网动漫久久久| 日韩一级欧洲| 在线观看视频一区| 久久久久国产免费免费| 亚洲网站视频| 亚洲人成绝费网站色www| 亚洲一区二区三区久久| 亚洲黄色av| 欧美一级视频一区二区| 夜夜嗨一区二区| 久久天堂成人| 久久久久国产精品一区三寸| 亚洲综合色自拍一区| 国产一区二区三区在线观看精品 | 久久精品噜噜噜成人av农村| 亚洲人成人一区二区三区| 久久网站热最新地址| 欧美日韩国产成人高清视频| 美玉足脚交一区二区三区图片| 欧美在线啊v| 亚洲一区视频在线观看视频| 免费一级欧美在线大片| 久久精选视频| 国产亚洲成人一区| 亚洲一区二区在线| 国产日韩专区在线| 一区二区欧美在线观看| 99精品久久久| 欧美日韩成人激情| 男人的天堂亚洲在线| 国产女精品视频网站免费| 亚洲第一久久影院| 激情久久婷婷| 久久久精品日韩欧美| 午夜亚洲福利| 日韩视频第一页| 在线观看日韩国产| 久久综合给合久久狠狠狠97色69| 亚洲电影在线免费观看| 久久国产视频网站| 久热精品视频在线观看一区| 国产一区二区三区四区五区美女 | 国产精品自拍小视频| 99视频精品全国免费| 亚洲视频1区2区| 欧美性猛交xxxx乱大交蜜桃| 午夜久久久久久久久久一区二区| 欧美一区二区三区播放老司机| 激情综合色丁香一区二区| 欧美激情精品久久久久久黑人| 欧美日本在线| 99亚洲一区二区| 亚洲欧美激情诱惑| 国产主播喷水一区二区| 久久人人爽人人爽爽久久| 亚洲三级性片| 欧美日韩一区在线观看| 中文成人激情娱乐网| 久久国产精品久久久久久久久久| 女女同性精品视频| 亚洲欧美日韩成人| 国产精品一区毛片| 久久夜色精品国产| 99ri日韩精品视频| 久久久久久久波多野高潮日日| 欧美屁股在线| 亚洲一区二区在| 欧美成人午夜视频| 国产亚洲一级高清| 欧美成人第一页| 蜜臀91精品一区二区三区| 亚洲人精品午夜在线观看| 国产精品va在线播放我和闺蜜| 欧美二区不卡| 国产亚洲欧美一区二区三区| 亚洲日韩成人| 久久不射网站| 9l视频自拍蝌蚪9l视频成人| 国产精品一区二区三区成人| 另类图片综合电影| 亚洲欧美国产高清| 亚洲国产欧美在线| 久久久久久亚洲精品杨幂换脸| 国产精品毛片a∨一区二区三区|国| 欧美成人午夜77777| 好男人免费精品视频| 亚洲一区二区免费| 亚洲福利视频二区| 久久天堂精品| 午夜综合激情| 亚洲网址在线| 99在线精品视频在线观看| 国语对白精品一区二区| 国产精品美女久久久久久免费 | 欧美中文字幕精品| 日韩午夜精品| 亚洲精品国精品久久99热| 国产日韩欧美综合| 国产精品美女久久| 欧美午夜宅男影院| 欧美日韩美女在线| 欧美女激情福利| 欧美大片在线观看| 久久亚洲视频| 国产精品实拍| 欧美日韩精品高清| 欧美电影资源| 西西裸体人体做爰大胆久久久| 黄色亚洲在线| 日韩午夜在线| 亚洲精品免费在线观看| 欧美日韩三区| 亚洲视频在线观看免费| 欧美与欧洲交xxxx免费观看| 99精品视频一区| 欧美在线视频一区| 一区二区三区高清视频在线观看| 99精品99久久久久久宅男| 精品91在线| 亚洲国产午夜| 91久久精品国产| 国产精品久久久久久久浪潮网站 | 亚洲少妇在线| 日韩亚洲一区二区| 欧美制服丝袜| 亚洲欧洲精品一区二区三区不卡 | 国产精品视频一区二区高潮| 欧美日韩在线观看视频| 欧美日韩久久久久久| 国产精品美女在线| 国产一区美女| 亚洲欧洲精品天堂一级| 99精品视频网| 性色av一区二区三区| 久久久之久亚州精品露出| 亚洲美女黄色| 亚洲一区二区高清| 久久久久一区二区三区四区| 麻豆精品网站| 欧美有码在线视频| 看片网站欧美日韩| 欧美在线播放一区二区| 久久久久一本一区二区青青蜜月| 一本大道久久a久久精二百| 国产日韩欧美视频| 欧美久久久久久久久久| 欧美视频二区36p| 激情欧美丁香| 亚洲特级毛片| 欧美成人精品一区二区| 一区二区日韩免费看| 欧美在线免费观看视频| 亚洲午夜一二三区视频| 欧美在线亚洲| 欧美三级乱码| 亚洲国产精品va在线看黑人动漫| 国产日韩欧美黄色| 亚洲免费观看视频| 久久免费偷拍视频| 亚洲视频在线观看视频| 久久综合伊人77777| 国产精品视频免费一区| 欧美体内谢she精2性欧美| 国产亚洲激情在线| 亚洲婷婷在线| 亚洲天天影视| 亚洲性夜色噜噜噜7777| 一本一本大道香蕉久在线精品| 国产在线国偷精品产拍免费yy| 欧美亚洲不卡| 亚洲欧洲一区| 欧美成人午夜免费视在线看片| 美日韩免费视频| 亚洲男人av电影| 欧美日韩免费观看一区二区三区| 欧美全黄视频|