大概六、七年前当我还在学 Asphyre 的时候,有看过一个以之编写的类似对对碰的“宠物对对碰”小游戏,虽然很简单,但我当时还是小小的沉溺过数个小时。而不久前,在闲逛论坛时无意看到了个以 FireMonkey 为所谓框架编写的另一款对对碰小游戏 GemGenie(需翻墙查看),由于其以 Delphi 开发,抱着看看 Delphi 现阶段对移动游戏开发的支持程度如何了,我下了个安装包,在我的华为荣耀U9508上跑了下。

  结果当然很失望,我很难比较流畅的游戏。这个 FireMonkey 出来已 5 年左右了,还是半个玩具。

  我自然比较倾向这个游戏的作者水平不会差(到哪去),所以不流畅的原因我想不归结于 FireMonkey 都难,而这玩意确实也一直备受诟病。

  当然,某司宣称过 FireMonkey 并非针对游戏开发而提供。

  那就让还用某司开发工具的人用某司的玩具工具去开发应用 APP 吧。

  大概七、八个月前,我在学 Unity3D 的时候想练练手,刚好那时候 Unity3D 已开始官方支持 2D 开发,所以那时我就用 Unity2D 重写了前文提及的“宠物对对碰”游戏。不过那时候我所学还非常浅,虽然确实重写出了这个游戏,但效果却很不尽如人意,如下:

PetGenie-LMLPHP

  像这样简单的游戏,如此多的 Draw Calls 显然是不可接受的(我应该正确使用 Sprite Packer 打包了图片资源)。

  那时我是通过 OnGUI 及 Graphics.DrawTexture 等来实现绘制等相关的,此做法明显还比较原始:

void OnGUI() {
if(Event.current.type.Equals(EventType.Repaint)) {
// 绘制背景
DrawBkgrd(); // 绘制剩余时间条
DrawTimeBar(); // 绘制得分
DrawScore(); // 绘制所有头像
DrawSpriteTextures(); // 绘制游戏信息
DrawGameCopyright();
}
} void DrawBkgrd() {
Graphics.DrawTexture(new Rect((Screen.width - cBkgrdW) / 2, (Screen.height - cBkgrdH) / 2, 386, 506),
bkgrdTexture, new Rect(0.0f, 0.0f, 1.0f, 1.0f), 0, 0, 0, 0, null);
} void DrawTimeBar() {
Graphics.DrawTexture(new Rect(41 + (Screen.width - cBkgrdW) / 2, 25 + (Screen.height - cBkgrdH) / 2, time / cTimeMax * 300, 10),
timeTexture, new Rect(0.0f, 0.0f, time / cTimeMax, 1.0f), 0, 0, 0, 0, null);
}

  

  这几天有些闲,我索性想重新实现下(绘制相关),于是开搞。

  ——游戏首先最好有个名字,既然有 GemGenie,那这个就叫 PetGenie(宠物精灵)吧!

  在重新导入及打包资源、废了些脑力回看及修改之前的代码后,得到了看起来似乎比之前的版本要优化的结果了:

PetGenie-LMLPHP

  这个版本看起来在绘制上应该不会有压力了:)

  导入手机,跑起来(如丝般顺滑),如下:

PetGenie-LMLPHP

PetGenie-LMLPHP

PetGenie-LMLPHP

PetGenie-LMLPHP

  这个版本里,以如下逻辑替代了之前版本的绘制相关代码:

void Update () {
if (!isGameOver) {
if ((!isMoving) || (!isUpdating)) {
timeLeft -= Time.deltaTime;
} if (timeLeft <= 0) {
isGameOver = true;
audioSrc.PlayOneShot(gameOverClip);
Application.LoadLevel("FailScene");
} if (score >= cScoreMax) {
Application.LoadLevel("WinScene");
}
}
else {
if (audioSrc.isPlaying) {
audioSrc.Stop();
}
} if (Input.GetKey(KeyCode.Escape)) {
isGameOver = true;
Application.Quit();
} // 检测输入信息
DetectInput(); // 更新精灵宠物信息
UpdateGenies(); // 更新剩余时间条大小&位置
UpdateTimeBar(); // 更新得分
UpdateScore();
} private void UpdateTimeBar() {
// 横向缩放
timeLeftTransform.localScale = new Vector3(timeLeft / cTimeMax, timeLeftTransform.localScale.y, timeLeftTransform.localScale.z);
// 改变 X 坐标(前一步的缩放是双方向内缩,所以这里要除以2)
timeLeftTransform.position = new Vector3(cTimeBarGapLeft + 5.12f * (timeLeftTransform.localScale.x - 1.0f) / 2, timeLeftTransform.position.y, timeLeftTransform.position.z);
}

  而这些逻辑的实现似乎有好几种方法,为简单起见(毕竟只是业余弄弄),我选了其中最简单的:摆控件(反正控件又不多)。

  这样的开发方式似乎也正是 Unity3D 的所长。

  这是最终的工程APK 文件。代码比较凌乱,整个设计也有很多改进之处,此仅为练手而已。

  注:

  1)、我手边没有比较好看的美术资源,也不会 PS作图,所以游戏看起来比较挫:)

  2)、这个网页是个很好的学习去处,包含 Uinty3D 及 Cocos 等相关资源(可能需要翻墙)。

05-04 05:57