道具

原文 Powerups
作者 JoeydeVries
翻译 包纸
校对 暂无

!!! note

  1. 本节暂未进行完全的重写,错误可能会很多。如果可能的话,请对照原文进行阅读。如果有报告本节的错误,将会延迟至重写之后进行处理。

Breakout已经接近完成了,但我们可以至少再增加一种游戏机制让它变得更酷。“充电”(译注:Powerups,很多游戏中都会用这个单词指代可以提升能力的道具,本文之后也会用道具一词作为其翻译)怎么样?

这个想法的含义是,无论一个砖块何时被摧毁,它都有一定几率产生一个道具块。这样的道具快会缓慢降落,而且当它与玩家挡板发生接触时,会发生基于道具类型的有趣效果。例如,某一种道具可以让玩家挡板变长,另一种道具则可以让小球穿过物体。我们还可以添加一些可以给玩家造成负面影响的负面道具。

我们可以将道具建模为具有一些额外属性的GameObject,这也是为什么我们定义一个继承自GameObjectPowerUp类并在其中增添了一些额外的成员属性。

  1. const glm::vec2 SIZE(60, 20);
  2. const glm::vec2 VELOCITY(0.0f, 150.0f);
  3. class PowerUp : public GameObject
  4. {
  5. public:
  6. // 道具类型
  7. std::string Type;
  8. GLfloat Duration;
  9. GLboolean Activated;
  10. // 构造函数
  11. PowerUp(std::string type, glm::vec3 color, GLfloat duration,
  12. glm::vec2 position, Texture2D texture)
  13. : GameObject(position, SIZE, texture, color, VELOCITY),
  14. Type(type), Duration(duration), Activated()
  15. { }
  16. };

PowerUp类仅仅是一个有一些额外状态的GameObject, 所以我们简单地将它定义为一个头文件,你可以在这里找到它。

每个道具以字符串的形式定义它的类型,持有表示它有效时长的持续时间与表示当前是否被激活的属性。在Breakout中,我们将添加4种增益道具与2种负面道具:

道具 - 图1

  • Speed: 增加小球20%的速度
  • Sticky: 当小球与玩家挡板接触时,小球会保持粘在挡板上的状态直到再次按下空格键,这可以让玩家在释放小球前找到更合适的位置
  • Pass-Through: 非实心砖块的碰撞处理被禁用,使小球可以穿过并摧毁多个砖块
  • Pad-Size-Increase: 增加玩家挡板50像素的宽度
  • Confuse: 短时间内激活confuse后期特效,迷惑玩家
  • Chaos: 短时间内激活chaos后期特效,使玩家迷失方向

你可以在下面找到道具的高质量纹理:

与关卡中的砖块纹理类似,每个道具纹理都是完全灰度的,这使得我们在将其与颜色向量相乘时可以保持色彩的平衡。

因为我们需要跟踪游戏中被激活的道具的类型、持续时间、相关效果等状态,所以我们将它们存储在一个容器内:

  1. class Game {
  2. public:
  3. [...]
  4. std::vector<PowerUp> PowerUps;
  5. [...]
  6. void SpawnPowerUps(GameObject &block);
  7. void UpdatePowerUps(GLfloat dt);
  8. };

我们还定义了两个管理道具的函数,SpawnPowerUps在给定的砖块位置生成一个道具,UpdatePowerUps管理所有当前被激活的道具。

SpawnPowerUps

每次砖块被摧毁时我们希望以一定几率生成一个道具,这个功能可以在GameSpawnPowerUps函数中找到:

  1. GLboolean ShouldSpawn(GLuint chance)
  2. {
  3. GLuint random = rand() % chance;
  4. return random == 0;
  5. }
  6. void Game::SpawnPowerUps(GameObject &block)
  7. {
  8. if (ShouldSpawn(75)) // 1/75的几率
  9. this->PowerUps.push_back(
  10. PowerUp("speed", glm::vec3(0.5f, 0.5f, 1.0f), 0.0f, block.Position, tex_speed
  11. ));
  12. if (ShouldSpawn(75))
  13. this->PowerUps.push_back(
  14. PowerUp("sticky", glm::vec3(1.0f, 0.5f, 1.0f), 20.0f, block.Position, tex_sticky
  15. );
  16. if (ShouldSpawn(75))
  17. this->PowerUps.push_back(
  18. PowerUp("pass-through", glm::vec3(0.5f, 1.0f, 0.5f), 10.0f, block.Position, tex_pass
  19. ));
  20. if (ShouldSpawn(75))
  21. this->PowerUps.push_back(
  22. PowerUp("pad-size-increase", glm::vec3(1.0f, 0.6f, 0.4), 0.0f, block.Position, tex_size
  23. ));
  24. if (ShouldSpawn(15)) // 负面道具被更频繁地生成
  25. this->PowerUps.push_back(
  26. PowerUp("confuse", glm::vec3(1.0f, 0.3f, 0.3f), 15.0f, block.Position, tex_confuse
  27. ));
  28. if (ShouldSpawn(15))
  29. this->PowerUps.push_back(
  30. PowerUp("chaos", glm::vec3(0.9f, 0.25f, 0.25f), 15.0f, block.Position, tex_chaos
  31. ));
  32. }

这样的SpawnPowerUps函数以一定几率(1/75普通道具,1/15负面道具)生成一个新的PowerUp对象,并设置其属性。每种道具有特殊的颜色使它们更具有辨识度,同时根据类型决定其持续时间的秒数,若值为0.0f则表示它持续无限长的时间。除此之外,每个道具初始化时传入被摧毁砖块的位置与上一小节给出的对应纹理。

激活道具

接下来我们更新游戏的DoCollisions函数使它不只检查小球与砖块和挡板的碰撞,还检查挡板与所有未被销毁的道具的碰撞。注意我们在砖块被摧毁的同时调用SpawnPowerUps函数。

  1. void Game::DoCollisions()
  2. {
  3. for (GameObject &box : this->Levels[this->Level].Bricks)
  4. {
  5. if (!box.Destroyed)
  6. {
  7. Collision collision = CheckCollision(*Ball, box);
  8. if (std::get<0>(collision))
  9. {
  10. if (!box.IsSolid)
  11. {
  12. box.Destroyed = GL_TRUE;
  13. this->SpawnPowerUps(box);
  14. }
  15. [...]
  16. }
  17. }
  18. }
  19. [...]
  20. for (PowerUp &powerUp : this->PowerUps)
  21. {
  22. if (!powerUp.Destroyed)
  23. {
  24. if (powerUp.Position.y >= this->Height)
  25. powerUp.Destroyed = GL_TRUE;
  26. if (CheckCollision(*Player, powerUp))
  27. { // 道具与挡板接触,激活它!
  28. ActivatePowerUp(powerUp);
  29. powerUp.Destroyed = GL_TRUE;
  30. powerUp.Activated = GL_TRUE;
  31. }
  32. }
  33. }
  34. }

对所有未被销毁的道具,我们检查它是否接触到了屏幕底部或玩家挡板,无论哪种情况我们都销毁它,但当道具与玩家挡板接触时,激活这个道具。

激活道具的操作可以通过将其Activated属性设为true来完成,实现其效果则需要将它传给ActivatePowerUp函数:

  1. void ActivatePowerUp(PowerUp &powerUp)
  2. {
  3. // 根据道具类型发动道具
  4. if (powerUp.Type == "speed")
  5. {
  6. Ball->Velocity *= 1.2;
  7. }
  8. else if (powerUp.Type == "sticky")
  9. {
  10. Ball->Sticky = GL_TRUE;
  11. Player->Color = glm::vec3(1.0f, 0.5f, 1.0f);
  12. }
  13. else if (powerUp.Type == "pass-through")
  14. {
  15. Ball->PassThrough = GL_TRUE;
  16. Ball->Color = glm::vec3(1.0f, 0.5f, 0.5f);
  17. }
  18. else if (powerUp.Type == "pad-size-increase")
  19. {
  20. Player->Size.x += 50;
  21. }
  22. else if (powerUp.Type == "confuse")
  23. {
  24. if (!Effects->Chaos)
  25. Effects->Confuse = GL_TRUE; // 只在chaos未激活时生效,chaos同理
  26. }
  27. else if (powerUp.Type == "chaos")
  28. {
  29. if (!Effects->Confuse)
  30. Effects->Chaos = GL_TRUE;
  31. }
  32. }

ActivatePowerUp的目的正如其名称,它按本章教程之前所预设的那样激活了一个道具的效果。我们检查道具的类型并相应地改变游戏状态。对于StickyPass-through效果,我们也相应地改变了挡板和小球的颜色来给玩家一些当前被激活了哪种效果的反馈。

因为StickyPass-through效果稍微改变了一些原有的游戏逻辑,所以我们将这些效果作为属性存储在小球对象中,这样我们可以根据小球当前激活了什么效果而改变游戏逻辑。我们只在BallObject的头文件中增加了两个属性,但为了完整性下面给出了更新后的代码:

这样我们可以通过改动DoCollisions函数中小球与挡板碰撞的代码便捷地实现Sticky效果。

  1. if (!Ball->Stuck && std::get<0>(result))
  2. {
  3. [...]
  4. Ball->Stuck = Ball->Sticky;
  5. }

在这里我们将小球的Stuck属性设置为它自己的Sticky属性,若Stikcy效果被激活,那么小球则会在与挡板接触时粘在上面,玩家不得不再次按下空格键才能释放它。

在同样的DoCollisions函数中还有个为了实现Pass-through效果的类似小改动。当小球的PassThrough属性被设置为true时,我们不对非实心砖块做碰撞处理操作。

  1. Direction dir = std::get<1>(collision);
  2. glm::vec2 diff_vector = std::get<2>(collision);
  3. if (!(Ball->PassThrough && !box.IsSolid))
  4. {
  5. if (dir == LEFT || dir == RIGHT) // 水平碰撞
  6. {
  7. [...]
  8. }
  9. else
  10. {
  11. [...]
  12. }
  13. }

其他效果可以通过简单的更改游戏的状态来实现,如小球的速度、挡板的尺寸、PostProcessor对象的效果。

更新道具

现在剩下要做的就是保证道具生成后可以移动,并且在它们的持续时间用尽后失效,否则道具将永远保持激活状态。

在游戏的UpdatePowerUps函数中,我们根据道具的速度移动它,并减少已激活道具的持续时间,每当时间减少至小于0时,我们令其失效,并恢复相关变量的状态。

  1. void Game::UpdatePowerUps(GLfloat dt)
  2. {
  3. for (PowerUp &powerUp : this->PowerUps)
  4. {
  5. powerUp.Position += powerUp.Velocity * dt;
  6. if (powerUp.Activated)
  7. {
  8. powerUp.Duration -= dt;
  9. if (powerUp.Duration <= 0.0f)
  10. {
  11. // 之后会将这个道具移除
  12. powerUp.Activated = GL_FALSE;
  13. // 停用效果
  14. if (powerUp.Type == "sticky")
  15. {
  16. if (!isOtherPowerUpActive(this->PowerUps, "sticky"))
  17. { // 仅当没有其他sticky效果处于激活状态时重置,以下同理
  18. Ball->Sticky = GL_FALSE;
  19. Player->Color = glm::vec3(1.0f);
  20. }
  21. }
  22. else if (powerUp.Type == "pass-through")
  23. {
  24. if (!isOtherPowerUpActive(this->PowerUps, "pass-through"))
  25. {
  26. Ball->PassThrough = GL_FALSE;
  27. Ball->Color = glm::vec3(1.0f);
  28. }
  29. }
  30. else if (powerUp.Type == "confuse")
  31. {
  32. if (!isOtherPowerUpActive(this->PowerUps, "confuse"))
  33. {
  34. Effects->Confuse = GL_FALSE;
  35. }
  36. }
  37. else if (powerUp.Type == "chaos")
  38. {
  39. if (!isOtherPowerUpActive(this->PowerUps, "chaos"))
  40. {
  41. Effects->Chaos = GL_FALSE;
  42. }
  43. }
  44. }
  45. }
  46. }
  47. this->PowerUps.erase(std::remove_if(this->PowerUps.begin(), this->PowerUps.end(),
  48. [](const PowerUp &powerUp) { return powerUp.Destroyed && !powerUp.Activated; }
  49. ), this->PowerUps.end());
  50. }

你可以看到对于每个效果,我们通过将相关元素重置来停用它。我们还将PowerUpActivated属性设为false,在UpdatePowerUps结束时,我们通过循环PowerUps容器,若一个道具被销毁切被停用,则移除它。我们在算法开头使用remove_if函数,通过给定的lamda表达式消除这些对象。

!!! important

  1. <fun>remove_if</fun>函数将lamda表达式为true的元素移动至容器的末尾并返回一个迭代器指向应被移除的元素范围的开始部分。容器的<fun>erase</fun>函数接着擦除这个迭代器指向的元素与容器末尾元素之间的所有元素。

可能会发生这样的情况:当一个道具在激活状态时,另一个道具与挡板发生了接触。在这种情况下我们有超过1个在当前PowerUps容器中处于激活状态的道具。然后,当这些道具中的一个被停用时,我们不应使其效果失效因为另一个相同类型的道具仍处于激活状态。出于这个原因,我们使用isOtherPowerUpActive检查是否有同类道具处于激活状态。只有当它返回false时,我们才停用这个道具的效果。这样,给定类型的道具的持续时间就可以延长至最近一次被激活后的持续时间。

  1. GLboolean IsOtherPowerUpActive(std::vector<PowerUp> &powerUps, std::string type)
  2. {
  3. for (const PowerUp &powerUp : powerUps)
  4. {
  5. if (powerUp.Activated)
  6. if (powerUp.Type == type)
  7. return GL_TRUE;
  8. }
  9. return GL_FALSE;
  10. }

这个函数简单地检查是否有同类道具处于激活状态,如果有则返回GL_TRUE

最后剩下的一件事便是渲染道具:

  1. void Game::Render()
  2. {
  3. if (this->State == GAME_ACTIVE)
  4. {
  5. [...]
  6. for (PowerUp &powerUp : this->PowerUps)
  7. if (!powerUp.Destroyed)
  8. powerUp.Draw(*Renderer);
  9. [...]
  10. }
  11. }

结合所有的这些功能,我们有了一个可以运作的道具系统,它不仅使游戏更有趣,还使游戏更具有挑战性。它看上去会像这样:

你可以在下面找到所有更新后的代码(当关卡重置时我们同时重置所有道具效果):