3.7其他辅助类
Helpers命名空间中还包括其他一些辅助类,大多数都像RandomHelper类一样简单,没有必要把它们都讲一遍,所以请你们自己看一下本章没有讲到的那些辅助类,如果想进一步了解它们可以使用其中的单元测试进行一些测试。
在介绍Breakout游戏之前,先简要地看看这样几个辅助类,它们在下面几章中会被反复地使用:SpriteHelper、EnumHelper和ColorHelper。
SpriteHelper类
上一章已经讲了很多渲染sprite的内容,当时为了配合单元测试而不得不使用一种简单的方式来处理sprite。很多时候需要把反复使用的有用代码放进可复用的类中,于是就产生了这样一个SpriteHelper类(如图3-10所示)。它提供了一个构造函数创建SpriteHelper实例,存储texture和Graphic Rectangle数据,还提供了一些渲染方法更方便地把sprite渲染输出到屏幕上,就像上一章中使用SpriteToRender类一样。
图3-10
这里的大多数方法的操作都不是很多,构造函数初始化变量的值,Render方法向sprite列表中添加SpriteToRender实例,RenderCentered方法在指定位置居中显示sprite,最后DrawSprites方法把所有sprite画到屏幕上。看一下其中的DrawSprites方法,它和前一章中的DrawSprites方法很像,不过有一些改进:public static void DrawSprites(int width, int height)
{
// No need to render if we got no sprites this frame
if (sprites.Count == 0)
return;
// Create sprite batch if we have not done it yet.
// Use device from texture to create the sprite batch.
if (spriteBatch == null)
spriteBatch = new SpriteBatch(sprites[0].texture.GraphicsDevice);
// Start rendering sprites
spriteBatch.Begin(SpriteBlendMode.AlphaBlend,
SpriteSortMode.BackToFront, SaveStateMode.None);
// Render all sprites
foreach (SpriteToRender sprite in sprites)
spriteBatch.Draw(sprite.texture,
// Rescale to fit resolution
new Rectangle(
sprite.rect.X * width / 1024,
sprite.rect.Y * height / 768,
sprite.rect.Width * width / 1024,
sprite.rect.Height * height / 768),
sprite.sourceRect, sprite.color);
// We are done, draw everything on screen with help of the end method.
spriteBatch.End();
// Kill list of remembered sprites
sprites.Clear();
} // DrawSprites()
{
// No need to render if we got no sprites this frame
if (sprites.Count == 0)
return;
// Create sprite batch if we have not done it yet.
// Use device from texture to create the sprite batch.
if (spriteBatch == null)
spriteBatch = new SpriteBatch(sprites[0].texture.GraphicsDevice);
// Start rendering sprites
spriteBatch.Begin(SpriteBlendMode.AlphaBlend,
SpriteSortMode.BackToFront, SaveStateMode.None);
// Render all sprites
foreach (SpriteToRender sprite in sprites)
spriteBatch.Draw(sprite.texture,
// Rescale to fit resolution
new Rectangle(
sprite.rect.X * width / 1024,
sprite.rect.Y * height / 768,
sprite.rect.Width * width / 1024,
sprite.rect.Height * height / 768),
sprite.sourceRect, sprite.color);
// We are done, draw everything on screen with help of the end method.
spriteBatch.End();
// Kill list of remembered sprites
sprites.Clear();
} // DrawSprites()
在调用该方法的时候,传递当前窗口分辨率的宽度和高度,并根据当前分辨率对所有sprite进行比例缩放,这对于支持Xbox 360的多分辨率非常重要。方法首先检查是否有东西要渲染,然后确保SpriteBatch类的静态实例(就是这里的spriteBatch变量)是否已被创建,调用Begin方法之后,在当前帧中遍历所有的sprite并重新调整它们的尺寸以适应当前屏幕的大小,最后当把所有sprite画到屏幕上之后再调用End方法。另外还要把sprite列表清空,为下一帧的渲染做准备。可以研究本章最后的Breakout游戏来看看这个类是如何工作的。
EnumHelper类
EnumHelper类(如图3-11所示)对于遍历枚举项以及获取枚举值的个数等操作非常有用。在Pong和Breakout游戏中没有用到任何枚举类型,但下一章的游戏在遍历block类型的时候使用Enum类(System.Enum)有很大帮助。注意,EnumHelper类用到了Enum类的几个方法,这些方法在.Net Compact Framework中并没有被实现。为了避免编译错误,在Xbox 360项目中通常排除整个EnumHelper类,不过在Windows平台上可以使用它。
图3-11
单元测试的TestGetAllEnumNames方法如下所示,该测试说明了GetAllEnumNames方法是如何工作的,它借助EnumHelper类内部定义的EnumEnumerator辅助类来遍历所有的枚举值。[Test]
public void TestGetAllEnumNames()
{
Assert.AreEqual(
"Missions, Highscore, Credits, Help, Options, Exit, Back",
EnumHelper.GetAllEnumNames(typeof(MenuButton)));
} // TestGetAllEnumNames()
public void TestGetAllEnumNames()
{
Assert.AreEqual(
"Missions, Highscore, Credits, Help, Options, Exit, Back",
EnumHelper.GetAllEnumNames(typeof(MenuButton)));
} // TestGetAllEnumNames()
GetAllEnumNames方法则使用了之前讨论的StringHelper类中的WriteArrayData方法:
public static string GetAllEnumNames(Type type)
{
return StringHelper.WriteArrayData(GetEnumerator(type));
} // GetAllEnumNames(type)
{
return StringHelper.WriteArrayData(GetEnumerator(type));
} // GetAllEnumNames(type)
ColorHelper类
ColorHelper类如图3-12所示,原本它还有更多的方法,但XNA中新的Color类比托管DirectX中使用的System.Drawings中的Color类的功能更加强大,所以有很多方法就不再需要了,不过它还是包含了一些对颜色操作非常有用的方法。
图3-12
例如,ColorHelper.Empty字段可以用来把shader效果参数设置为空值——0,0,0,0通常不是有效颜色值,它是完全透明的,而黑色的Alpha值是255。/// <summary>
/// Empty color, used to mark unused color values.
/// </summary>
public static readonly Color Empty = new Color(0, 0, 0, 0);
/// Empty color, used to mark unused color values.
/// </summary>
public static readonly Color Empty = new Color(0, 0, 0, 0);