成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

【精品教程】Cocos2d-x v3.6制作射箭游戲(二)

移動開發 游戲開發
本章我們的主要任務是創建射箭的弓箭手(也就是游戲豬腳),并且讓這個豬腳隨著觸摸點的改變不斷的旋轉手中的弓箭。

本章我們的主要任務是創建射箭的弓箭手(也就是游戲豬腳),并且讓這個豬腳隨著觸摸點的改變不斷的旋轉手中的弓箭。

分析:

對于這個射箭的角色而言,它能不停的射出弓箭。當我們按住屏幕上某點時,會從該角色拿弓箭的手的位置“畫”一條標注箭支運動軌跡的紅線(看似拋物線);當在屏幕上滑動手指或鼠標時,這條紅線會隨著觸摸點的位置不停的變換軌跡;當松開屏幕上的手指或鼠標時,會射出一支弓箭,這支弓箭會按最終的紅線路徑移動。另外,玩家手中的弓箭會隨著屏幕上的手指或鼠標旋轉。

Player 類

下面我們一起來創建這個 Player 豬腳類,其初步定義如下:

 
  1. class Player: public Sprite 
  2.     public
  3.     Player(); 
  4.    
  5.     bool init(Vec2 playerPos); 
  6.     static Player* create(Vec2 playerPos); 
  7.    
  8.     void createPlayer(); 
  9.     void createPlayerHpBar(); 
  10.     void rotateArrow(Point touchPoint); 
  11.     void createAndShootArrow( Point touchPoint); 
  12.     void shootArrow(); 
  13.     void finishRunAction(); 
  14.     void update(float dt);       
  15.    
  16.     CC_SYNTHESIZE(int, playerHp, PlayerHp);        // 玩家血量值 
  17.     CC_SYNTHESIZE(bool, startDraw, StartDraw);     // 是否開始畫紅色的路徑線 
  18.     CC_SYNTHESIZE(bool, isRunAction, IsRunAction); // 玩家是否正在執行射箭動畫 
  19.    
  20.     private
  21.     Vec2 playerPos;            // 角色在 tmx 地圖上的位置 
  22.     Size playerSize;           // 角色尺寸 
  23.     Size winSize;              // 屏幕窗口尺寸 
  24.     Sprite* playerbody;        // 角色身體 
  25.     Sprite* playerarrow;       // 角色的弓箭,也就是會隨觸摸點旋轉的弓和箭部分 
  26.     Sprite* hPBgSprite;        // 角色血條背景精靈 
  27.     ProgressTimer* hpBar;      // 角色血條 
  28.     ccQuadBezierConfig bezier; // 路徑貝賽爾 
  29.     DrawNode* drawNode;        // 這里表示我們的線條對象 
  30.    
  31. }; 

以上的各方法都是我們這兩章需要實現的,其他更多的方法我們將在后面需要的時候再擴充。 

其中CC_SYNTHESIZE宏的作用是定義一個保護型的變量,并聲明一個getfunName函數和setfunName函數,你可以用getfunName函數得到變量的值,用setfunName函數設置變量得值。如:CC_SYNTHESIZE(int, playerHp, PlayerHp);定義了一個整型的 playerHp 變量,同時還聲明了 getPlayerHp() 和 setPlayerHp() 兩個方法。 

ccQuadBezierConfig是我們新定義的一個結構體,后面我們會詳細的講解。

下面我們就從上到下依次來看看以上的各方法。

創建角色

首先是 Player 的初始化(init)和創建(create),這里我們通過給定 Player 的位置來創建該角色,而這個傳入的坐標位置應該是我們從 TiledMap 的對象層中讀取到的位置(上章有講)。具體代碼如下:

 
  1. Player * Player::create(Vec2 playerPos) 
  2.     Player *pRet  = new Player(); 
  3.     if (pRet && pRet->init(playerPos)) 
  4.     { 
  5.         pRet->autorelease(); 
  6.         return pRet; 
  7.     }else 
  8.     { 
  9.         delete pRet; 
  10.         pRet = NULL; 
  11.         return NULL; 
  12.     } 
  13. bool Player::init(Vec2 playerPos) 
  14.     if (!Sprite::init()) 
  15.     { 
  16.         return false
  17.     } 
  18.     this->playerPos = playerPos; 
  19.     createPlayer();        // 創建角色 
  20.     createPlayerHpBar();   // 創建角色血量條 
  21.     scheduleUpdate(); 
  22.     return true

下面我們接著來看看 createPlayer 方法,該方法將初始化我們的 Player 角色,代碼如下所示:

  1. void Player::createPlayer() 
  2.     playerbody = Sprite::createWithSpriteFrameName("playerbody.png"); 
  3.     playerSize = Size(playerbody->getContentSize().width/2, playerbody->getContentSize().height / 3*2);   
  4.     // 設置Player的尺寸,大小略小于playerbody的尺寸,這樣利于我們后面更準確的進行碰撞設置。 
  5.     playerbody->setAnchorPoint(Vec2(0.7f, 0.4f)); 
  6.     this->addChild(playerbody); 
  7.     this->setPosition(Vec2(playerPos.x+ GameManager::getInstance()->getObjectPosOffX(), playerPos.y + playerSize.height * 0.4f)); 
  8.    
  9.     playerarrow = Sprite::createWithSpriteFrameName("playerarrow.png"); 
  10.     playerarrow->setPosition(Vec2(00)); 
  11.     playerarrow->setAnchorPoint(Vec2(0.3f, 0.5f)); 
  12.     this->addChild(playerarrow);    

createPlayer 方法中我們將創建如下所示的一個游戲角色。

201506251005113376.jpg

因為沒有找到合適的游戲資源(原游戲中得到的資源都是零件,要使用需要把它們一幀一幀重組),所以我們的游戲一切從簡,不整那些復雜的。 

這里我們只把角色簡單分成了兩個部分,第一部分當然是玩家的身體playerbody,第二部分是隨著觸摸點/鼠標旋轉的手和弓箭playerarrow。(PS:當然因為資源限制這個原因,可能會稍稍降低咱游戲的檔次,應該不能怪我啰!O(∩_∩)O~)

設置playerbody位置時,你可能已經發現,我們并沒有把角色身體設置在傳入的playerPos處,而是對它稍微做了一定的調整。這是因為我們傳入的位置它是緊貼本格瓦片底部的(我們制作tmx文件時,需要這樣做。上章沒說清楚,這章補起,要記住哦!)。如下圖所示:

201506251006122519.jpg

Y值坐標也不可太接近本格瓦片底部,也就是不要設為9.990,9.998這類太接近10的,因為 tmx 文件中存放的坐標值是整數,如果設為9.990,9.998,那么存放的值會是9.990 X 32 = 319.68 = 320,同理 9.998 X 32 也是 320。320 對于瓦片大小是32 X 32的地圖來說是個特殊的數字,因為 320 /32 = 10。這樣在程序中就會誤以為9.990,9.998之類的點是坐標上的第10個點。

而且上章我們也說過,由于分辨率適配的原因,對象組中對象的位置與實際的位置是有一定的偏差的,所以我們在設置角色身體位置時,需要修正這些偏差。 

以上代碼中設置位置的原理圖如下:

201506251008052690.jpg

其中,對象組在 X 軸上的偏移值我們把它保存在了 GameManager 中,而 GameManager 是個單例類,后面章節我們會詳細的講解。當然如果你現在就想運行代碼,那就先把GameManager::getInstance()->getObjectPosOffX()部分去掉吧。

創建好角色后,接下來我們需要創建角色的血量條,血量條可通過 Cocos2d-x 中封裝好的進度條類 ProgressTimer 來創建。其代碼段如下:

  1. void Player::createPlayerHpBar() 
  2.     // 創建血條底,即進度條的底背景     
  3.     hPBgSprite = Sprite::createWithSpriteFrameName("hpbg.png"); 
  4.     hPBgSprite->setPosition(Vec2(playerbody->getContentSize().width / 2, playerbody->getContentSize().height)); 
  5.     playerbody->addChild(hPBgSprite); 
  6.     // 創建血條  
  7.     hpBar = ProgressTimer::create(Sprite::createWithSpriteFrameName("hp1.png")); 
  8.     hpBar->setType(ProgressTimer::Type::BAR); // 設置進度條樣式(條形或環形) 
  9.     hpBar->setMidpoint(Vec2(00.5f));        // 設置進度條的起始點,(0,y)表示最左邊,(1,y)表示最右邊,(x,1)表示最上面,(x,0)表示最下面。 
  10.     hpBar->setBarChangeRate(Vec2(10));      // 設置進度條變化方向,(1,0)表示橫方向,(0,1)表示縱方向。 
  11.     hpBar->setPercentage(100);                // 設置當前進度條的進度 
  12.     hpBar->setPosition(Vec2(hPBgSprite->getContentSize().width / 2, hPBgSprite->getContentSize().height / 2 )); 
  13.     hPBgSprite->addChild(hpBar); 
  14.     hPBgSprite->setVisible(false);   // 設置整個血條不可見,我們將在Player 遭受攻擊的時候再顯示血條。 

#p#

旋轉角色弓箭

接下來我們來讓 Player 的弓箭部分跟隨著觸摸點/鼠標旋轉。所以我們定義了如下的函數:

  1. void Player::rotateArrow(Point touchPoint) 
  2.     // 1     
  3.     auto playerPos = this->getPosition(); 
  4.     auto pos = playerPos + playerarrow->getPosition(); 
  5.     // 2 
  6.     Point vector = touchPoint - pos; 
  7.     auto rotateRadians = vector.getAngle(); 
  8.     auto rotateDegrees = CC_RADIANS_TO_DEGREES( -1 * rotateRadians); 
  9.     // 3 
  10.     if (rotateDegrees >= -180 && rotateDegrees <= -90){ 
  11.         rotateDegrees = -90
  12.     } 
  13.     else if (rotateDegrees >= 90 && rotateDegrees <= 180){ 
  14.         rotateDegrees = 90
  15.     } 
  16.     // 4 
  17.     auto speed = 0.5 / M_PI; 
  18.     auto rotateDuration = fabs(rotateRadians * speed); 
  19.     // 5 
  20.     playerarrow->runAction( RotateTo::create(rotateDuration, rotateDegrees)); 

rotateArrow方法的參數為觸摸點的位置。

1)獲取角色弓箭在游戲場景中位置;

2)計算弓箭的旋轉角度。 

這里利用三角正切函數來計算,原理如下圖所示: 

201506251009138662.png

vector(offX,offY) 是觸摸點到弓箭之間的向量,通過 getAngle 方法,我們可以得到 vector 向量與X軸之間的弧度。 

再者,我們需要把弧度 rotateRadians 轉化為角度,CC_RADIANS_TO_DEGREES就是能把弧度轉化為角度的宏。轉化時乘 -1 是因為Cocos2d-x中規定順時針方向為正,這與我們計算出的角度方向相反,所以轉化的時候需要把角度a變為-a。

3)控制旋轉角度的范圍,即只讓它在角色右半邊內旋轉。

4)計算弓箭旋轉時間。 

speed表示炮塔旋轉的速度,0.5 / M_PI其實就是 1 / 2PI,它表示1秒鐘旋轉1個圓。 

rotateDuration表示旋轉特定的角度需要的時間,計算它用弧度乘以速度。

5)讓弓箭執行旋轉動作。

觸摸響應

好了,現在 Player 就初步定義好了。接下來,我們回到游戲場景把Player加入進去,并來測試下弓箭是否跟隨觸摸點旋轉。

在 Cocos2d-x 3.x 引擎中,實現觸摸響應的流程基本是一致的。所以在 3.6 中,其過程依舊是:

  • 重載觸摸回調函數;
  • 創建并綁定觸摸事件;
  • 實現觸摸回調函數。

所以我們要測試弓箭是否跟隨觸摸點旋轉,第一步請先在 GameScene 中重寫如下的觸摸回調函數,并聲明變量:

  1. virtual bool onTouchBegan(Touch *touch, Event *unused_event);  // 開始觸摸屏幕時響應 
  2. virtual void onTouchMoved(Touch *touch, Event *unused_event);  // 觸摸屏幕并在屏幕上滑動時響應 
  3. virtual void onTouchEnded(Touch *touch, Event *unused_event);  // 觸摸結束時響應 
  4.    
  5. private
  6.     Point preTouchPoint;      // 上一個觸摸點 
  7.     Point currTouchPoint;     // 當前觸摸點 

接著,我們需要在 GameScene 的 init 初始化函數中創建并綁定觸摸事件,并先隨便創建一個 Player 對象,用于測試。如下:

 
  1. SpriteFrameCache::getInstance()->addSpriteFramesWithFile("texture.plist""texture.pvr.ccz");   
  2. player = Player::create(Vec2(winSize.width / 4, winSize.height/5));  
  3. this->addChild(player); 
  4.    
  5. // 獲取事件分發器 
  6. auto dispatcher = Director::getInstance()->getEventDispatcher(); 
  7. // 創建單點觸摸監聽器 
  8. auto listener = EventListenerTouchOneByOne::create(); 
  9. // 讓監聽器綁定事件處理函數 
  10. listener->onTouchBegan = CC_CALLBACK_2(GameScene::onTouchBegan,this); 
  11. listener->onTouchMoved = CC_CALLBACK_2(GameScene::onTouchMoved,this); 
  12. listener->onTouchEnded = CC_CALLBACK_2(GameScene::onTouchEnded,this); 
  13. // 將事件監聽器添加到事件調度器 
  14. dispatcher->addEventListenerWithSceneGraphPriority(listener,this); 

Player 的位置是固定的,我們當然不能隨便設,這里只是為了測試。后面的章節中我們會創建一個類來專門管理從 TiledMap 中得到的對象,包括Player、敵人、道具,磚塊等。

以上 plist 和 pvr.ccz文件是我們的打包資源,它們是用 Texturepacker 編輯器打包而來。更多詳細內容請點此查看

綁定好觸摸事件后,最后我們需要實現它們,代碼如下:

 
  1. bool GameScene::onTouchBegan(Touch *touch, Event *unused_event) 
  2.     currTouchPoint = touch->getLocation(); 
  3.     if( !currTouchPoint.equals(preTouchPoint)){ 
  4.         player->rotateArrow(currTouchPoint); 
  5.     } 
  6.     preTouchPoint = currTouchPoint;   
  7.     return true;   
  8.    
  9. void GameScene::onTouchMoved(Touch *touch, Event *unused_event) 
  10.     currTouchPoint = touch->getLocation(); 
  11.     if( !currTouchPoint.equals(preTouchPoint)){ 
  12.         player->rotateArrow(currTouchPoint); 
  13.     } 
  14.     preTouchPoint = currTouchPoint; 
  15.    
  16. void GameScene::onTouchEnded(Touch *touch, Event *unused_event) 
  17.     // 射箭,下章內容 

在 onTouchBegan 和 onTouchMoved 函數中,處理方法是一樣的。即當當前觸摸點與之前的觸摸點不一致時,就旋轉 Player 的弓箭。 

getLocation 方法將 touch 對象中保存的屏幕坐標轉換成我們需要的 Cocos2d 坐標。 分不清屏幕坐標和Cocos2d 坐標的童鞋請參考Cocos2d-x3.0坐標系詳解一文。

當觸摸結束時,Player 對象需要射出弓箭,這個我們暫時不寫。

運行游戲,此時你就可以看到想要的效果了。關于本章資源,請點此下載

責任編輯:倪明
相關推薦

2015-07-17 10:38:21

教程COCOS射箭游戲

2011-12-12 10:40:08

Cocos2d-X游戲開發開發環境

2013-05-22 15:49:46

2013-12-03 10:58:50

Cocos2D-X磚塊地圖

2012-04-17 12:58:44

Cocos2D-X

2012-04-17 12:38:46

cocos2d-x

2012-02-19 20:10:23

Cocos2d-x fCocos2dWindows Pho

2014-08-13 10:07:02

游戲引擎

2012-04-17 12:44:38

cocos2d-x

2013-04-16 10:02:47

cocos2d-x懶人Android開發

2014-04-11 11:10:14

Cocos2d-x v手游引擎

2013-06-03 17:04:20

CocoStudioCocos2D-X添加CocoStudi

2013-05-22 14:38:44

iOS開發Cocos2d-x坐標系統

2014-04-21 14:58:27

觸控Cocos2d-x觸控科技

2012-04-17 12:52:01

cocos2d-x

2014-07-31 16:57:30

2012-04-17 12:47:27

cocos2d-x

2012-04-17 10:06:08

cocos2d-x

2012-04-17 10:59:31

cocos2d-x

2012-04-17 13:09:13

Cocos2d-x
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 日韩精品一区二区三区中文在线 | 羞羞的视频在线看 | 在线免费国产视频 | 日韩久久久久久 | 免费簧片视频 | www.夜夜骑| 久久国产一区 | 午夜免费看视频 | 欧美国产精品久久久 | 欧产日产国产精品国产 | 亚洲伊人精品酒店 | 亚洲精选久久 | 狠狠的操| 免费福利视频一区二区三区 | 国产欧美一区二区三区日本久久久 | 91高清视频在线 | 日韩手机在线看片 | 国产一区二区久久 | 亚洲国产精品一区 | 日日夜夜天天综合 | 亚洲欧美精品 | 国产日韩欧美一区二区在线播放 | 国产一级片久久久 | 成人不卡一区二区 | 99久久精品国产一区二区三区 | 一本一道久久a久久精品综合 | 一级黄色片日本 | 91九色porny首页最多播放 | 精品九九在线 | 欧洲免费毛片 | 伊人99| 国产成人精品一区二区三区在线 | 亚洲精品久久久蜜桃 | 国产aⅴ精品 | 亚洲综合精品 | 国产日韩久久久久69影院 | 国产不卡一区 | 另类亚洲视频 | 欧美精品一二三 | 国产久视频 | 久久久久亚洲 |