搜索
亲,你还没有登录哦,马上登录。如果还没有帐号,请立即注册X
查看: 2217|回复: 6

GBA游戏制作菜鸟入门篇

[复制链接]
  • TA的每日心情
    别闹
    2013-12-11 16:57
  • 签到天数: 129 天

    [LV.7]王国居民III

    发表于 2011-9-13 09:21:20 | 显示全部楼层 |阅读模式
    今天怎么样也要把它转过来,虽然那些工具都下载不了了,可能时间久了吧,
    6 N: [  G" ^" ]9 ]( l( z下面为转载内容:5 R3 }8 Z( |. A& U
    [转帖]GBA游戏制作菜鸟入门篇: a2 x7 {7 u8 ]  c  O' y
    此文章由混沌星辰雷之精灵撰写
    9 B: g# J; ?9 s原文链接:http://gba.gamego.cn/dispbbs.asp ... ID=10024&page=1" x2 g7 _  R: m- ~; x
    ——————————————————————————————————————————7 h; P; y7 b, F6 O! [( k0 h/ A
    转帖目的:希望大家对GBA游戏热爱的同时,也来了解下GBA开发过程,其实很简单,只要用心,你也可以做出好的GAME) o% W. M) n. u8 }6 Z' X% X/ v+ _
              目前转的是所有菜鸟入门篇,如果回了这个入门篇,做一个类似逆转类型的游戏也就比较轻松了~' v. W: X6 x8 A/ j' x' y$ b
              很简单,  大家试试
    ; d9 K# T! Q  {# U( o# j+ ]/ ^——————————————————————————————————————————
    & C1 C3 T! I& Y( F. B最近见到论坛上有新手帖我的教程,这本来是一件好事,但是想不到那位兄弟竟然忘了帖图!我觉得鄙人教程虽然不算是什么好东西,但是图片,还算得上是画龙点睛之笔。那兄弟的帖子我不能编辑,我只能在这里重新帖一遍教程,并尽量编辑成最容易阅读的格式,放在这里供大家参考吧!7 h$ O: k6 W3 ?- [8 l
    ( O5 i" ^& B* C0 `6 b$ l' D' e( E8 z/ u
    下面是正文了:
    6 e4 w2 n) s# j8 }1 S' w8 j1 ?9 B1 ~4 c2 j7 b" O$ K5 L
    GBA游戏编程菜鸟入门篇(一)——简介及开发工具( P# O2 ?$ m5 o; y: v

      ?+ W3 ^* s& k玩GBA游戏编程的时候,发现GBA编程不难。自己总结了一些心得,写成教程,贴在这里吧!! q% n+ Z, N% N& n

    # k( L+ X& k5 I8 @& k8 W5 T本来这是要投稿给《掌机迷》的,但是人家迟迟没有回音,我不耐烦了,放在这里免费供大家浏览。
    + @8 E, l& D% ^( i
    ( ]9 ]( n3 h8 s; A' X2 F编程的时候发现,GBA编程涉及很多GBA硬件方面的知识,也和微机原理有着千丝万缕的联系。我前些日子曾经入门过GBA游戏汉化,当时不知道或者懵懂的东西,现在差不多都豁然开朗了。如果你有汉化游戏的欲望,又对微机原理略微了解一些,而苦于自己是菜鸟一直徘徊在汉化门口,那么你不妨看看我的教程,相信你会有所发现。如果你对游戏开发颇有兴趣,又略微学过一些C语言,希望看到自己在GBA上面的作品,而看到天书一般的汇编代码吓得六神无主,那么你也不妨看看我的教程,相信你会很快找回自信……8 a* z& m( d2 U* B  ~! E, L

    7 e+ r' Z6 p$ f2 n8 U5 N5 R, g# N好,闲话少说,开始!
    5 `2 K) Q( L8 g  d* Z" q9 U
    5 c( `( T1 p- L3 A* H首先声明一下,本人水平很低,低到什么程度呢?我没有学过C语言;我只学过3年的VB。因此,我个人对C语言并不是很了解。如果我的教程中出现可笑的低级错误,请各位海涵……另外,也请编程达人多多指正,雷之精灵在这里感谢各位!
    9 T) }, L1 k0 z" \  p0 f1 g! |$ F/ |4 @$ n1 U) {
    许多人认为编程很难。没错!编程是很难。但是,所谓世上无难事,兴趣是最好的老师。只要你有兴趣,有持之以恒的决心,那么就不难。我在论坛上见到不少朋友非常喜欢玩游戏,也曾梦想着自己编写自己的游戏;但是看到涉足编程领域困难重重就一直在门口徘徊。如果你也和他们一样,那么请放下包袱,尝试着走进来;或许你会发现,编程领域也不是像你想象的那样……
    7 u& l% A  o9 X4 I8 }! v/ B
    / }/ M" P- J# c, a7 c0 n& w: W关于在GBA上面编程,现在网上的达人们似乎都认为它要比在PC上面编程简单一些。我不持这样的观点。毕竟,在GBA上面编程,迄今为止还没有像VB、VC之类的所谓“所见即所得”的面向对象的开发工具;一切都要靠汇编和C语言……如果你接触过汇编或者C语言,那么,上手就很容易了!
    4 `' q' R' ?" n5 p4 m" h2 X  U# M& V2 Q
    好了,闲话少说,开始吧!由于我不知道大家对于计算机语言的了解情况,在这里,我就假定你是已经稍微接触过C语言,但是绝对没有深入研究过,大约也就只能写出“求三个数的最大值”之类的小程序;对于微机原理处于懵懂状态,能看得懂16进制,知道所谓“高字节数据存放在内存高地址,低字节数据存放在内存低地址”的菜鸟……门槛不高吧?
    ' z8 m) n) r  `( H1 ~3 a
    ; z) \/ T* Z. y, ]4 ?3 f, V由于使用C语言要比使用汇编效率高,所以,在这里我建议大家使用C语言。我这里,也将以C语言来讲述。- P4 w5 h/ t& {/ x% b) W

    % g. J, j; i3 d6 \) D工欲善其事,必先利其器。必要的工具,我们必须具备:
    / e" w: A# e, j# ^+ V0 @0 G
    8 s7 m" A' f2 u9 s+ P; s4 z1.函数库:这里,我强烈推荐使用“GsLIB”。这是一种高效率的GBA函数库,作者是国内开发达人黄炎中。GsLIB将GBA的各种功能进行封装,形成标准库函数。使用的时候直接调用。现在可以下载到的版本有2.6 ARMSDT版本、2.6 GNUPRO版本和2.7 GNUPRO版本。在我的教程里面,使用2.6 ARMSDT版本。不同的版本有着一些细小的差异。GsLIB可以在黄炎中达人的主页http://gsgba.yeah.net直接下载到。完全免费!如果你打算以我的教程为指导,那么请下载2.6 ARMSDT版本。7 A+ [) t$ _. j0 L* S1 W
    ( G- n# c- ]9 w% i9 [
    2.开发环境和编译器:有两种编译器可以选择,就是ARMSDT和GNUPRO。其中,ARMSDT是ARM公司的商业软件,要想使用得要交钱的;而GNUPRO是免费软件,任何人都可以直接下载到。不过,GBA的CPU是ARM公司的产品,所以使用ARMSDT开发效率高。而GNUPRO据说是有BUG,开发效率也比ARMSDT低,生成的ROM体积也稍微大一些……无论如何,由于我们使用的GsLIB是ARMSDT版本,所以必须使用ARMSDT。但是这个软件,网上似乎很难下载到……我在这里也没法提供下载(软件安装包接近30M,附件方式放不上来)不过,最近我有发现D版的ARMSDT,是ARMSDT2.50版本,试用版,在这里如果只是当作GBA编译器的话可以正常使用。在搜索引擎里面找一下,应该可以找到吧……如果实在找不到,论坛的群里有共享。但是,群里的ARMSDT,有不能用的先例……(6lay大人:(阴沉着脸)你说谁?)! @% |0 `# Q) A  Q$ e
    2 U' c4 g. W+ X1 p# ?
    3.写程序的工具:既然是C语言,那么,在VC里面写就当然是最好!要是没有,你也完全可以在“记事本”或者“写字板”里面写代码,写完之后保存成后缀名为.c的文件。不过,这两个Windows自带的工具都有一些缺点,比如不能显示行号,不能自动缩进,函数名也不能彩色显示之类的。在这里我就要极力推荐UltraEdit了!这是个16进制编辑器,也可以用来写文本文件。它的功能之强大,你用了就知道了!这个软件,网上多得满天飞。估计你任意找个下载软件的网站都能下载到……9 z8 H5 ~8 _1 L3 |  C+ I

    0 Z# C7 M( |' f4.测试工具:写完程序,经过编译链接之后,生成的文件就是我们想要的ROM了。它的后缀名是.gba……这个,我不说你也知道用什么打开吧!在这里我要说明的是,VBA的版本不要太低,因为有些功能在低版本的VBA里面还没有开放。另外,记住要用英文版,中文版的话,和教程里面会出现不一致,你也许会不知所措。8 h. r$ P; ]3 m8 a( v# S. p6 v
    6 n& p7 P3 e  {
    5.调试工具:你写出来的ROM出现BUG甚至错误是必然的。调试是一项很关键的任务。这里需要一个强大的反汇编模拟器。我推荐的是No$GBA。这也是商业软件,据说正式版需要2000美元……但是我们可以很得意的找到破解版……在稍微大一点的模拟网站或论坛里面应该可以找得到。不过,据说现在在网上流传的No$GBA,声音没有破解,因此我们编译出来的ROM,用这个模拟器打开,如果有背景音乐的话会出现噪音。不过这也无所谓,菜鸟教程中声音方面基本不需要调试……就算是要调试,也可以在VBA里面调试……9 N: k# H) C) a( W3 N( x, i6 i
    5 e$ Z% c: Q# \  E5 c& j- ^
    6.组合精灵编辑器:这是GsLIB的辅助工具之一,不知为什么黄达人没有把它和函数库放在一起。你需要单独下载。下载地址也是http://gsgba.yeah.net。或者到TGB论坛去下载:http://www.tgb.name/index.php?act=Attach&type=post&id=573
    ( d8 c. m2 Z( z3 j
    : b, p% q2 I7 P2 }6 Y剩下的差不多就是辅助工具了……! r, G6 i) v8 W( F$ t

    . `$ M9 Y2 N. X& }* F! P1.PhotoShop:基本上PS就是为了处理图片的调色盘才被我加进去的。游戏中最常用的就是图片,而在GBA里面,我们需要关心图片的调色盘。PS可以很轻松的处理调色盘。……你不知道什么叫做调色盘?不急不急,用到的时候我自然会讲解!PhotoShop的下载地址不用我说吧?找不到的话去附近卖光盘的商店转转,100%有卖的!
    + i4 A8 U2 ^  y; I
    5 w, g; A7 Y/ l2.ModPlug:如果你想要在你的游戏中加入背景音乐或者是真人语音,那么就少不了这个软件!GBA不能直接支持MID,虽然现在有个所谓的“MP2K”可以使GBA播放MID但非常不幸的是我们的编译器ARMSDT不支持MP2K!所以这条路就走不通了……幸运的是GBA支持MOD。什么是MOD?我也不知道……反正你就认为这是一种类似于MID的音乐格式就行了……到时候我会详细讲述如何用ModPlug制作MOD,如何在GBA上使用MOD的……这个软件也是外国人写的,界面是全英文,好像还没有汉化版,凑合用吧!下载地址我不说了,去搜索引擎里面搜索一下,一定会找到的。) Y7 r8 M9 Q4 M7 P. }. v

    / Z$ {/ c9 v1 z; v9 S$ X4 V# Z需要的工具大概就这么多了。当然,系统自带的工具也是少不了的!“画图”、“计算器”之类的,随手就到!/ d/ w  p  L7 ?0 }9 U0 C! j
    2 G% a: q2 i+ k4 A
    那么,今天就到这里。从下次开始,我们就要正式进入编程。下次要讲的内容是:在GBA上面显示文字!理论课内容是“中断请求和响应”。! D% n6 C. ]6 j- W

    - f4 y9 U6 P* y8 D6 A$ ]" U, R
    7 r  o! r+ M6 b! j& E$ x5 B7 Z3 s% ^! K5 e3 E
    GBA游戏编程菜鸟入门篇(二)——中断请求及显示文字; J, M+ [% }0 [) a. I6 U1 e

    & p' V7 n! _1 r5 v5 M& t  u1 F7 T' A嘿嘿……大家好!又见面了!上次教程中提到的必备工具,有没有准备好啊?有准备的,那就开始吧!
    5 M$ G* r* |7 d/ D5 F% s7 c0 y2 P1 B3 n  }- @8 m+ i
    GBA的显示屏分辨率大家都知道吧!240*160。就是说,宽240像素,高160像素。" i3 a' z, D# J2 w
    % N  I  V; n% g0 q2 h) h/ H
    GBA上有6种显示模式,分别命名为MODE0到MODE5。这六种模式中,MODE0、1、2很相似,被称为TILE方式;MODE3、4、5也很相似,被称为位图方式。9 u0 [% K# X  h
    ( d3 h/ h$ R- B- @8 H
    什么叫做“TILE方式”呢?简单地说,GBA上面的图像,是用一块一块的小方块拼成的。这种小方块叫做“TILE”,大小是8像素*8像素。在MODE0至MODE2中显示图片,就以这种方式显示。你想不想直观的看一下什么是TILE?找一个规模稍微大一点的GBA汉化网站或论坛,搜索一下叫做“TLP”的软件。下载下来之后,用它打开任意一个GBA的ROM。注意必须是GBA的ROM,也就是说不要压缩。打开之后,你会看到一大片花花绿绿的什么东西。这些花花绿绿的东西,被网格分开。其中每一小格都是8*8像素。这一个小格,就是一个TILE。
    / \; e$ q0 B- W* p3 g. V- Z7 _5 \9 u' \

    5 T- o& [+ e$ C7 W2 d* }7 |: i& u, |( x! d* E4 `$ j; t
    那么“位图方式”就很简单了。就是所谓的“点阵图像”。MODE3至MODE5下面以这种方式显示图像。
    ) [2 ~9 a1 L+ k
    # L: U9 a# `' Y0 s( a  c) _但是MODE4和MODE5还和MODE3有点不同……MODE4和MODE5采用了一种叫做“双缓冲”技术,使得显示高速的图像不会出现闪烁。双缓冲,形象地说,就像两张叠在一起的图片,上面那张你能看的到,下面那张被上面的挡住看不到。当你看到了上面那张之后,下面那张翻上来,上面那张翻到下面换成另一张图片。这样你就看到了刚才在下面现在又翻到了上面的那张图片。然后再将这张图片翻到下面,把那张换过的图片再翻上来,同时再将翻下去的换图片……反正就像这绕口令一样,看到上面的图片的同时将下面的图片更新,然后翻转,使你能看到下面那张图片;你看到翻上来的图片的同时更新下面的图片……再翻转……; {; k% O7 N& N( R/ w$ B

    0 d& Y0 t; j: C7 C) A7 J' `那么,在哪种方式下可以显示文字呢?哪种都可以!但是,不同的方式下显示文字的函数有点不同。我们先试着写个GBA的DEMO来显示文字。/ m4 x0 ], R) }0 Q5 Y
    # z/ j8 l7 w6 v8 m
    在哪种方式下显示呢?TILE吧。游戏经常用这种模式显示文字。3 `# G" _5 @% ]

    6 o1 S5 Y9 I% V  I; ~* h6 [所有的软件安装好了吗?还没有?得得的!我来!, r# u* J5 X4 P

    9 ^# e& N2 d: V0 q3 X+ k首先安装UltraEdit。方便起见简称UE。UE是准绿色软件,双击安装文件,你随意设置安装路径。安装完之后桌面上就有快捷方式了。如果你下载的版本是试用版,去网上找注册码,或者掏腰包注册。两种途径随你选择。9 L) t1 q: Z  R3 g  T1 \/ Y; q
    # y* y! M5 a# I! s
    然后安装ARMSDT。双击setup.exe进行安装,自己随意选择路径,然后你可能会看到下面这个对话框。$ W1 s6 `6 B) |5 _# k& K$ V

    , g( u7 g9 b$ N, h& c. }- M3 [8 z* G7 J4 d+ N" T( ]. b
    6 N, }" q; h: ?. `6 A9 W+ m* A

    # e0 R: h% H/ [; u4 k5 H# ?
    $ G# n; D. h6 H' s, C6 ]: |2 Y1 Q: s3 W1 x+ m

    ! q, D! M3 b; O2 o7 ?( P5 t全都不选。点击Next。然后可能会弹出一个错误对话框。7 _$ l7 S+ }  w+ m2 X
      @  u0 q3 o" n1 N5 E( F
    不要管它,继续。(这个对话框在有的系统上不会出现。系统是WinXP的时候不会弹出来,但Win2000会弹出来。不过这也不是绝对的,我同学的电脑,XP系统,也会弹出来……反正就是,管它弹出还是不弹出,继续就行了……)
    3 I3 ~. O# U: \( g( f  l7 q7 m' c- r5 G
    然后小等一下就搞定了。安装结束的时候可能会询问你是否要建立快捷方式,不要!建立了也没用!ARMSDT仅仅是一个编译器,除了编译我们用不着它的其他任何功能!这里我假设你是把ARMSDT安装到了D:\\ARMSDT文件夹下面。安装正确的话,文件夹下面会出现文件夹“BIN”、“LIB”、“INCLUDE”、“TEMPLATE”和文件“DeIsL1.isu”、“MFC42.DLL”、“MSVCRT.DLL”、“README.TXT”、“UNINST.DLL”。" w8 S- s7 v' X' A. a3 M6 R

    # s+ k4 o' _7 q) R然后就是函数库GsLIB了。这是绿色软件,解压缩到任意文件夹。这里我假设你是把GsLIB解压缩到了D:\\GsLIB文件夹下面。解压正确的话,文件夹下面会出现文件夹“DOC”、“INCLUDE”、“LIB”、“SAMPLES”、“SYSTEM”、“TOOLS”和文件“cgo.exe”“gsgo.bat”。' `/ l+ j' D" r5 C

    1 S% I: [+ }0 J( |! o9 n这里我有必要解释一下GsLIB的这些文件和文件夹:0 H8 [+ S6 V9 w$ e. Y

    4 A. V8 a% R$ O首先是DOC文件夹。里面是GsLIB的帮助文件。非常详细!唯一有点缺憾的是解释的不是很明白。无论如何,这里是学习GsLIB的天堂。以后在我的教程里面,如果出现了你没有见过的函数,就到这里面找吧。, |! H7 L' G5 W* v( T. |8 X
    2 y% m9 n' U6 d" g% J
    再者是Include文件夹。里面是库函数。如果你打算看更低层的东西例如寄存器和内存单元的分配以及命名,那么这里面或许有你想要找的。
    % K# N% X. e  a9 m( L9 M; R7 d! I( a$ d, K
    然后是Lib文件夹。我不知道这里面的那个文件有什么用……
    8 R- e3 r, O  G7 H
    8 C; M; r. E3 Z9 s5 z- I  tSAMPLES文件夹!这是最令人激动的文件夹。里面有不少的例子,基本上都很简单!你一定要看!以后我们写程序,就在这个文件夹里面写。* F2 q+ o, k. `1 J6 T1 E8 U

    " K" h; A, V' u& R: o6 L6 ZSYSTEM文件夹。这是GsLIB的系统文件夹,我们会用得到的。2 @3 u& K8 d2 N, s" x# R, T

    - y. z4 I% F7 ?* a' xTools文件夹。这里面是开发游戏的时候,有可能用得到的工具。% O' R/ j, V6 \( r" s
    , J- J# c' I' X/ ^1 j" h* y4 z
    然后是文件:cgo.exe。这个文件不用管它。
    7 ~! b& n; Z/ b7 n
    2 @! Q* {% ]9 `! X1 RGsgo.bat。批处理文件。相当重要的文件!你马上就知道,编译程序的时候,就靠它了。
    3 L. I. k3 r: \. V9 |& v3 C1 m8 t
    行了,继续!5 I/ B! x& X. q( o$ w

    0 W/ S* F( @; I# ~( X4 |模拟器们,都是绿色软件,随你安装吧!6 ]. f$ Z- H; k8 h* @
    ! I, L& B6 B8 M+ U
    那几个辅助工具,我就不讲解了。想必你们都会了!9 a: b8 o) U! V6 p

    ! m" Z+ w+ b$ E  F/ c哦!差点忘了!你有下载“组合精灵编辑器”吧!这是GsLIB的专用工具,将它解压缩到GsLIB\\Tools文件夹里面。% O) q! |4 w: v, h
    # f- I  R* E5 l7 P, U8 q9 F3 x
    先进入GsLIB\\system文件夹,复制“makefile”文件。
    . J; v- k# r9 y& I4 h5 ]9 z
    5 @1 u3 J8 ]) m2 R5 V然后退出来进入GsLIB\\samples文件夹,新建一个文件夹。假设你建立的文件夹叫做“Temp”。
    # k2 b; ?+ x9 Y) m! X9 Q! B  B7 s" e$ g2 z6 Q- W8 O
    进入Temp文件夹,粘贴,这个时候刚才复制的“makefile”就粘贴到了这里。$ z) |$ Q0 v" U* A
    : n2 E+ D% y. m( k: j1 J% f8 ]! l8 V
    再新建一个文本文件,取名叫“main.c”。记住后缀名必须是“.c”!这个时候文本文件的图标会变成UE图标的样子。双击,UE自动将这个文件打开了。- H$ u) p. z' m/ R3 A) O
    . t) V& v& ~8 z6 j$ g8 `
    然后,开始写代码吧!- x8 W; M1 X* Y8 e5 a
    ) N9 \, ?1 _: j+ f6 ]# s
    从现在开始,我会详细解释所有的代码。里面涉及的函数,在GsLIB\\DOC文件夹里面的帮助文件里有详细的说明。  y& y9 ^. e& x

    " L, _# f3 {  L5 _* t#include <GsGBA.h>//这是载入GsLIB函数库。以后你写的所有程序,第一句必须加上这个包含。9 C2 [8 ?1 P* N  M# W. W, ]

    : O1 q4 }4 L8 l#include <GsText.h>//要想显示文字,必须载入字库。以后你写的程序如果有需要显示汉字,记住加上这个包含。: q1 Z0 F, m5 i4 f3 y' k( ^. s. `

    & C: L: o3 T3 }3 N& Dconst IntrFuncp IntrTable[14]=) D  l8 L2 E4 g. V3 W4 I( t( `3 w
    {
    6 V, A2 B$ h6 E& Y5 o, Q   dummy,dummy,dummy,dummy,* V8 r! N$ v! }: Y
       dummy,dummy,dummy,dummy,( s7 V# z5 h& z, e: z% w
       dummy,dummy,dummy,dummy,- j- u% f. Z$ e* m4 G
       dummy,dummy# q& J8 A3 h) n* f/ z8 j
    };//这是中断向量表。下面我会简单讲解中断,现在你就记住要这样写就行了。% ]- H( p+ ]# {$ m2 `4 Q
    % Q5 p3 n/ O. q' F  @
    int AgbMain()//主函数。注意这和标准C不一样!标准C中主函数是Main();,而这里是AgbMain();。
    1 l) [: s& e$ Y7 l; }, M% L0 U
    ' S* {. b0 k4 I0 L: s, z+ u{
    , y9 C- n! G6 R% l! W1 U9 {5 p0 Z, @# ?
    4 l2 Z7 `* @( E3 C  GsSetMode( MODE_0);//设置模式函数。这里将模式设置为MODE0,TILE模式。
    & \9 ]1 X$ \. U/ ^' P5 {( o% g2 ?. j, ]% s7 X; x
      BGPaletteMem[1]=0xffff;//这是设置调色盘。BGPaletteMem[ ]可以看作是一个数组。这句就是说,将16进制数0xffff写入数组BGPaletteMem的第1号元素中。以后,调色盘中中的第1号颜色就是它了。3 t1 m$ X( r+ v& V. I6 y6 _
    / l& w+ f' _4 n- g, u1 S5 r
      GsInitBG();//初始化BG。这是一个很重要的函数。只要在TILE模式下,就要涉及一个“BG”问题。只要使用BG,则必须在使用之前初始化BG。关于BG的详细内容,下一次的教程《GBA游戏编程菜鸟入门篇3——TILE模式及显示图片》中我将以大篇幅详细讲解;再下一次的教程《GBA游戏编程菜鸟入门篇4——多层BG及TILE模式下的多层BG》我将以更大篇幅讲解。现在,可以说,掌握了BG,就掌握了GBA游戏编程的三分之一……4 C6 O+ b! `6 r3 q9 K

    4 Q0 b- g8 w8 Q- g, C1 t- N  GsSetBg(0,Gs_BG_TEXT_SIZE_256x256,Gs_BG_COLOR16,0,21,0);//设置BG。如果想对BG进行什么操作的话,则必须在初始化BG之后对其进行设置。根据帮助文件,可以得知这是设置BG0,大小是256*256,TEXT属性,16色,TILE数据在第0号区域,MAP数据在第21号区域,不允许马赛克特效。现在你或许对这些参数为什么要这样设置感到疑惑,不过,下一次的教程你就明白了。如果下一次的教程你还没明白,那么再下一次的教程你就能明白。再次强调,BG是很重要的,掌握了BG,就掌握了GBA编程的三分之一……熟练运用BG,你就可以自由自在地在GBA上操纵图片显示。
    7 H, n& Y6 t, z; X5 j8 Z2 T0 h( }0 E' t0 J1 z$ Z2 T
      GsTileTextInit(0,3,21,0);//初始化文字显示。如果你想在GBA的某个BG上显示文字,则必须对该BG进行初始化文字显示。根据帮助文件,这是将BG0进行初始化,MAP数据在第21号区域,使用第0组调色盘。
      n8 w3 z8 f: k! A
    / `" q5 T7 M3 V  GsSetBgState(0,ON);, q8 q, i0 V. e

    1 o: a1 M2 I6 J# `$ z4 j9 V  GsSetBgState(1,OFF);
    7 M6 I6 R8 [: R0 s0 ]5 Y. P" O3 z, r3 p
      GsSetBgState(2,OFF);, A' P6 H  ?+ x- h; r

    - V+ }% u  H, A  `6 X  GsSetBgState(3,OFF);//打开BG0,关闭BG1~3。在MODE0模式下,BG有4层。在这里我们只用到了BG0,并用于显示文字。所以,将BG0打开,其他BG,关闭就行了。
    " Z+ ^1 C6 Q3 K8 M! X' j8 ?' j  GsTileTextOut(20,40,"我爱TGB,我爱烧录地带!",1);//显示文字啦!这就是传说中的显示文字的函数。根据帮助文件,这是在GBA屏幕上的(20,40)坐标处开始显示“我爱TGB,我爱烧录地带!”这几个字,颜色是调色盘中第1号颜色。在程序的开始,我们已经定义了调色盘中的第一号元素是0xffff,这里就使用这种颜色。
    2 T, d# g4 B2 V& h" N$ q0 s- L* X7 E( W1 S1 \& @
    while(1)//开始循环。
    2 \# v% r7 }1 ?7 H  {6 c3 u2 g8 l4 c2 H; z. e
    {6 U, S* X2 X. W* I% r3 y& v

    . e6 c) K9 J8 n7 e/ S' }4 }    GsWaitSync();//等待同步。这是什么意思呢?就是说等待发生VBLANK中断。每当GBA显示完成一幅图片之后,就发生一次VBLANK中断。也就是说,当发生中断之后,屏幕上的图片已经显示完了。( i; t' Y/ y& k9 ~
    }//GBA的ROM就是一个永不停止的循环,在这里进行循环,就是显示完文字之后,等待同步。一旦等到了同步,文字也显示完了。于是屏幕上就在坐标(20,40)处显示出了“我爱TGB,我爱烧录地带!”这几个字。颜色为0xffff,就是白色。
    ) u2 [2 b; t8 ]7 c: freturn 0;//函数值返回主函数。由于主函数是int类型而不是void类型,所以会有返回值。, s' j: r% K! _, Z
    }7 \% a: \/ E/ h# W0 y% X7 W
    3 y3 t, Q) B- [0 ]% N' x; @
    就这样,写完了。保存。很难吗?去掉那些注释看看,很简单!+ D2 ~( e! B2 ]$ v! @3 Q
    ' [; _& d, i" n+ s
    程序写完了,编译。
    1 B* Z- w2 Z" _# K6 D& E4 k! j$ m7 e
    退到GsLIB文件夹下面,双击gsgo.bat文件。这个时候会弹出来一个DOS窗口。) M! ~4 ~6 q$ m/ B
    , r! u- M$ U+ j

    5 g& E' U: A' |! j1 W" N+ w* `- [8 l

    / T0 g' _  Q7 Z* T0 x/ x5 t
    9 Y6 B1 ~$ A8 A5 Q/ w4 L1 d
    ! C0 K4 a' P% G( N- |键入“cd samples\\Temp”回车,进入你新建的那个文件夹。
    # p( M: Q# V) ~' m, n$ C# Q& S3 U( }2 j7 T& a
    然后键入“make”。make在这里是编译的意思。一点回车,窗口立即向下刷屏。
    ' N: J& z+ ]  R" r9 s. C
    5 a2 l  W# F8 g这就说明编译链接成功。注意看下面有句话“ARM Linker:finished, 0 informational, 1 warning and 0 error messages.”这里标明了你的程序里面的警告错误。这个时候再进入GsLIB\\samples\\Temp文件夹看看,是不是多了几个文件?其中有个叫做“main.bin”的文件,那就是ROM啦!把它的后缀名改成.gba,然后用VBA打开看看吧!9 Z0 ^, c; ^- T& V7 t; w
    8 M2 H4 s! h$ _8 F- X6 I( o
    2 J3 g, K1 T8 `

    5 E; D: }% U* L$ H3 L# X# L( a4 Q! c  }' f" d% C
    至于“ARM Linker:finished, 0 informational, 1 warning and 0 error messages.”这句话,我们需要关心那个warning和error。如果出现的是“1 warning and 0 error messages”,就是说一个警告,没有错误。这一个警告是什么呢?你看看多出来的文件里面,是不是有一个“log.txt”呢?这就是记录warning的文件。打开看看,大意就是在Windows环境下,生成的文件未必能正常运行。这是当然的啦!GBA的ROM是运行在GBA平台的……又不是Windows平台!而模拟器,只是在Windows平台上模拟了一个GBA平台而已……如果出现的warning不止一个,那就说明你的程序里面有未定义的函数什么的,编译没通过……而要是出现error,那就说明程序有错误,无法编译了!这两种情况一旦出现,仔细检查程序,改正遗漏错误什么的……" v2 Q5 k! f8 O1 ]2 u6 {, T
    4 f* f4 M) q8 G" K8 y
    我刚才说到make这个命令。这个命令在DOS里面是没有的,只有你安装了ARMSDT,make命令才可以使用。在这里make是用来编译程序,生成ROM的。还有两个命令,我也有必要说一下:就是make clean和make vba这两个命令。make clean是清除所有文件,包括ROM,但是保留程序文件main.c。当你的程序里面有新添加资源的时候,编译之前最好用这个命令清除一下然后再编译;因为有时候程序有添加新资源但是其他一些编译产生的中间文件比如o文件、s文件什么的还是保持没有添加资源的状态,这个时候编译出来的ROM会出错。make vba,就是编译完成之后立即调用VBA载入ROM察看。不过遗憾的是它调用的VBA是SDL版本,这个版本没有图形界面,很难操作,不推荐使用。/ C6 ]6 C2 ?: I8 l

      z3 H$ O1 Q8 x/ \最后,该讲解中断了。
    7 k& G! v# T) ~5 K. P4 ~) m0 c' d( r/ j7 E  t1 Y- @6 q
    中断是CPU为了协调外设和它之间速度和反应的差异才设立的。CPU的运行速率非常高,相比之下外设就相当慢了。假如CPU想访问键盘,而这个时候你并没有按键,难道CPU就一直等着你按键吗?当然不会这样!这个时候CPU继续它的工作,当你有按键的时候,键盘向CPU发出按键中断请求,也就是请求CPU中断程序,处理你按下的按键。当CPU收到键盘的中断申请时,暂停程序,转去处理键盘。处理完键盘之后,CPU再回到刚才中断的程序处,继续运行……
    ( Q3 W# K% k- h& W* s* ^) v/ S% X  P" X9 J9 d* g
    GBA的部件中,能发出中断的部件很多,那么,这么多中断,CPU怎么判断到底是谁发出的中断请求呢?中断向量表!中断向量表中有14个中断可以受到你的控制。这14个中断,依次是:, |) g* D* b8 ?, I4 N4 {! ]

    7 @0 c8 ?2 J7 C8 f" g$ GVBLANK中断、HBLANK中断、VCOUNTER中断、TIMER0中断、TIMER1中断、TIMER2中断、TIMER3中断、串行通信(Serial Communicating)中断、DMA0中断、DMA1中断、DMA2中断、DMA3中断、按键(Key)中断还有卡带(Cartrage)中断。9 u( I4 f' T& Y% |' n  r
    ! f0 P9 d) u: N. ]  r; ^# C
    这14个中断中,常用的只有两个:VBLANK中断和DMA0中断。你会看到以后我的程序中只用一个VBLANK中断就完全够了。关于DMA0中断我一般不用,估计只会在讲DMA的时候才会用到吧……
    5 R! k3 y6 T/ g* m* p
    4 A9 ^, N: w* o6 M8 k怎么使用中断呢?如果你的程序很简单,不需要中断,那么就像上面写的一样,14个中断全部填“dummy”空函数,或者干脆不填,留个分号在那里就行;如果你想让程序响应VBLANK中断,那就在中断向量表中VBLANK的位置替换成你的中断服务子程序。具体怎样做呢?下次我就给出讲解和例子!下次要讲的内容:在GBA上显示图片!理论课内容是TILE模式详解。下一次的教程中,我们就详细看看TILE模式,并对BG进行初步解释。一般的GBA游戏中,TILE模式下的MODE0是最常用的。学好了TILE模式,或者说,学好了MODE0的使用,你基本上就可以自由操纵GBA了。位图模式,相信你可以独立学习……
    0 V! ~# I6 K7 X

    4 R2 ?6 Y/ R* Q
    6 v; e* ]4 s3 p- g6 L- x
    6 z5 k+ f6 P( J" L
    1 H' v3 P0 ^+ y# e% w  }! a% v9 H8 Q* R- q% [- ^
  • TA的每日心情
    别闹
    2013-12-11 16:57
  • 签到天数: 129 天

    [LV.7]王国居民III

     楼主| 发表于 2011-9-13 09:22:30 | 显示全部楼层
    GBA游戏编程菜鸟入门篇(三)——TILE模式及显示图片% x, V( D; I6 B6 _1 k) j1 H* k
    ; H! p/ d1 Y6 _
    今天我们来看一下GBA怎么显示图片。+ C5 a6 \0 x8 o8 N9 \# F

    + f7 J5 E' G  B) u8 f! T* H  Z( h- r还记得TILE模式吧!GBA有两种显示模式——TILE模式和位图模式。其中MODE0、1、2属于TILE模式,MODE3、4、5属于位图模式。9 g# H% h0 s1 [! R, |1 v
    3 ]8 U0 J6 j) j& J
    那么,一幅图,或者精确地说,位图,包含哪些数据呢?TILE、MAP、PALETTE。3 M: f8 s( M# h! [+ z
    4 ]6 V. E5 \: i
    TILE:就是前面曾经说过的,8*8像素的小方块。整张图片就是由这样一块一块的方块拼成的。
    1 P8 d, U" C8 O& d  `$ a, d: c  \- w7 t9 x( _+ S& g0 c, k" H
    MAP:一块一块的小方块拼成了图片,那么,哪块小方块在图片的什么位置,怎么确定呢?MAP数据就用来确定TILE在整张图片中的坐标。还包括其他一些数据这里我暂时不做涉及。
    - }% {, Z( c. w
    % H6 F3 V- i9 I3 L+ {- |/ Y9 Q+ G1 T- RPALETTE:调色盘。就是图片中所有用到的颜色。GBA支持的调色盘有3种:16色(4位)、256色(8位)和32768色(15位)。可见,对于真彩的24位位图(1600多万色),GBA是不支持的。不过,对于人类来说,恐怕没有谁能分辨出来32768色和24位色的区别吧!所以,支持这三种调色盘,可以说足够了。/ Y0 s+ I/ g/ g, V# i, t- \
    & M8 _' i3 i% V; N

    / O4 c  P. N1 e2 y' M+ b! H! e% ~# Q& J3 h6 o. S

    * X3 c9 `6 }' ~) A2 y) x- P那么,你就会明白了,一张256*256像素的图片,包含(256/8)*(256/8)=1024个TILE。这1024个TILE,形成了一个TILE数组。$ P) t& |/ ~3 ~7 @* Z; r3 ]

    ) q9 H; }+ x: N9 o+ B. w7 f那么,确定了图片的三个重要数据,GBA怎么显示这张图片呢?图层的概念你听说过吧!为了使图像富有立体感,GBA上面应用了图层的技术。你在GBA上面看到的图片,那就是图层叠加技术产生的效果。比如下面这张图:
    8 K  g, D8 G0 {! S7 y) a4 p. U- }* H+ e( t

    ) n5 t! A* C& Z, |# p' }# F% x: c9 r* S8 v5 z: ~3 S' o$ c
    7 h* o$ Z" A; \

    5 `/ h3 d! G+ M+ `) v9 W它是怎么做出来的呢?它是由下面这几个图层叠加出来的!
    " s( [$ S9 Q/ V% z7 E
    : n' J2 s5 f# z
    1 H! Z& @) F  e, y. [& _; ]1 @& n5 s1 `# X

    * y3 D" ]) i5 b- Y$ g" y8 R
    , s" C& h' d' V+ g" Q4 N; t$ g1 ^3 s: W* W4 L& |
    假如我规定,在这里黑色是透明色,就是说如果图片中有某部分是黑色,那么在GBA上这部分就呈现透明状态不显示。这样的话将这几幅图叠在一起,显示出来的就是上面那幅图。当然,叠加是有顺序的。从上面那张完整的图中能看出来,“口袋怪兽”这一层应该是最上层,因为它下面的几层被它覆盖了一部分。烈空座这层应该是最底层,因为它似乎哪层都没能覆盖……/ d4 D* W5 K+ I% ?  M

    9 U$ p* S' @7 ?  x. C1 i(众人:你等等!)雷精灵:有什么疑问吗?(众人:你截的图是不是少了一幅啊?那“绿宝石”“PUSH START BUTTON”和公司版权那几个字,属于哪一层?)
    4 u3 a1 t9 W, i/ f0 N, f0 v2 W$ ^% |; Z
    雷精灵:我正要解释这个问题:那几个字,不是图层。它们是“精灵”。它们处于图层的最上方。有关精灵这部分,以后我会详细讲解。精灵的出现,堪称游戏中的一大革命。掌握了这部分,你就能作出很漂亮的动画效果了!
    " d) Z2 ~/ \. E5 w+ ?. O& T7 ]
    9 W$ s& u0 n; p; a" mGBA有几个图层可用呢?4个。分别被称为BG0、BG1、BG2和BG3。这4个图层,GBA本身并没有规定谁是高层谁是底层,而是完全由你规定。每一个图层,你都可以随意操纵,例如图层面积、使用的调色盘、特效设置等等。8 R( M; R  \- V

    8 E& Q4 R" m/ W7 E  i$ _+ S' @" o$ B# Z有了图层的概念,那么下面的就很容易理解了。
    8 |! l9 V+ T& A# J# a: f4 x% b
    : n5 J# D) ^) M2 h/ L0 b; e+ X8 WTILE方式下,你知道,有三种模式。游戏中最常用的是MODE0,因此我重点说明MODE0。其他几个MODE,因为都是TILE方式,大同小异你自己完全可以理解!
    ( e! Z$ c/ A* |4 ?* N0 Q8 o! I; N  Z& h: ^
    MODE0,具体是什么样的呢?
    5 k9 L' D% {. M' ~6 }4 ]
    8 p0 p: I$ B8 w8 ?) @( g: d  zMODE0由4个非旋转(TEXT)图层组成。是普通游戏里用的最多的模式。硬件上支持的大小有这么几种:256*256、256*512、512*256和512*512。你会发现这几种大小都超过了GBA的显示区域。惊讶?别,GBA硬件最大支持1024*1024大小的图片;而在GsLIB的支持下,理论上可以支持无限大的图片。要这么大的BG有什么用?一般的RPG里面的地图,不会就只有一个GBA屏幕那么大吧!这个时候GBA通过滚动BG来显示整张图片。这种技术叫做“卷轴”。
    # u" j7 D2 b/ i/ u& ]9 Z1 x/ }2 G$ p7 Y! H7 U0 _- o0 V
    MODE0支持两种调色盘:16色(4位)和256色(8位)。也就是说,MODE0下面的这4个图层,可以都是16色的,也可以都是256色的,也可以是既有16色也有256色的。
    0 V  R6 \2 w; c! i/ X$ H
    & Y9 t& W: M# X3 JMODE0不支持图层旋转。旋转,不用多解释吧!你要是想做出图层旋转特效,不要使用MODE0。具体一些……《马里奥赛车》玩过吧!杯赛刚开始的时候,伴随着背景音乐,画面由远拉近,同时镜头由正对选手们的方向转向背对选手们的方向。那些选手是“精灵”我们暂时不讨论;背景,那就是“旋转”!不过,这个“旋转”,还不是真正的图层旋转,这实际上是传说中的“MODE7”。关于这方面的技术我也不是很了解……现在我们先放一下,继续回到MODE0。  X: K% ?3 T* J4 o/ n; p$ R! @
    " P4 n" S! B# x- P& D
    嗯……就这么多啦。其他几种MODE,我给个表格比较。0 u( B+ N. u4 C7 P" A8 d; O* b4 t. e

    . g. j1 ^9 r( k* e! m$ S- f" p' ]2 @- x+ n7 x

    0 r" z$ n) I' {: m( r& a* g+ A$ V* M; \. Z2 `. k

    0 D5 V% I# L" N
    1 w/ p, \: v7 T# ~% T! B明白了这些理论,下面这个程序你就不难理解了。我们现在就写个在MODE0下显示图片的DEMO。这一次,我们将让CPU能够响应键盘操作。什么效果呢?还记得MODE0下面BG的大小均超过GBA屏幕大小吗?我们在键盘上按方向键,对BG进行卷轴。3 c) o/ Y7 @0 N. [" F- n
    & W& R' D7 _  a& t9 f
    和上次一样,首先,在GsLIB\\samples\\文件夹下面新建一个文件夹;把GsLIB\\system文件夹里面的“makefile”复制到新文件夹里面;新建一个叫做“main.c”的文本文件,用UE打开……准备工作就完成了。我这里还是假设你新建的文件夹叫做temp。
    " K( U+ r' E- B
    3 |$ [9 p% y5 P' q让GBA显示什么图片呢?新建一个位图,起名叫“BG.bmp”,然后用Windows的“画图”打开,将图片的大小设置成256*256像素,然后你随便画点什么。保存。+ [4 ]0 V# `; K& g2 L

      l8 O* Q0 M0 m
      P. B: P8 d# c9 g0 U# d
    4 x- w$ \: x  g6 X% c. S  N9 I- o! Z& W: L: C0 V$ v
    然后用PS将这个位图转成16色的。# x& x4 m( G3 l" C- }

    ! x* t/ `  a. r- F# {) u  K(众人:你等等!)雷精灵:有什么疑问吗?(众人:转成16色?为什么要转成16色?)雷精灵:还记得MODE0支持的两种调色盘吗?这次我们让GBA在MODE0下面显示16色位图。(众人:那么,假如说我的位图颜色很丰富,转成16色岂不就会掉色很严重?)
      j' H9 O9 f4 i  o0 e( D
    ! L% a2 r, R  {. K$ U' M7 q雷精灵:没错!16色对于高画质的的图片的确很不适合,因为颜色太少,会使画质严重下降。这个时候你就用256色。但是256色依然是颜色太少,如果你对画质要求非常高,比如是想做CG特写之类的,那MODE0就无能为力了。这个时候你就用位图方式下的MODE3吧!不过,256色一般情况下也应该足够了。你可以用PS将一个真彩的位图转成15位位图另存,然后再转成8位位图另存,你会发现不仔细看基本上也没多少差别……更何况是在GBA的屏幕上,也不怎么能看得出来……3 Q( S) x! o; S/ t, X( B
    ( B8 r4 r' Z0 S& ^, t8 j
    好,图片准备好了。这张位图,大小是256*256像素,颜色为16色。格式为bmp。
    ( M  B/ G. j7 |- _+ T  |6 m8 ]% R, C- s% `
    但是GBA并不能直接识别位图,我们需要把位图转成三种数据——TILE数据、MAP数据和PALETTE数据。
    ; o: |3 m  f6 \$ @
    . L3 j2 {( C0 f, P双击GsLIB文件夹下的“gsgo.bat”,弹出DOS窗口,键入“cd samples\\temp”,进入你新建的那个文件夹。(众人:还没写程序编译干什么?)雷精灵:谁说要编译了?3 P7 J4 O! \# \7 n, o+ x

    * P" s. L7 s* |2 d7 g7 v键入下面的命令:“gfx2gba -fsrc -c16 -t8 -mm -pBG.pal BG.bmp”,点回车之后,文件夹下面多出三个文件来,分别是“BG.pal.c”“BG.map.c”和“BG.raw.c”。这几个文件,想必你一眼就能看出来是什么吧!还不知道?用UE打开这些文件,你就完全明白了。; N4 z. L9 O! L0 d# c
    $ M+ ~) q" q5 m2 s1 F1 g  m
    (众人:那个命令是干什么的?)雷精灵:gfx2gba,这是GBA用的一个工具。它可以将一张位图转化成位图的三个数据,并保存成C语言数组。关于这个工具的详细介绍,GsLIB的帮助文件里面有很详细的说明。不过,请注意一下,这次我们转化的图片,大小是256*256的。这不是凭空规定的,也不是因为MODE0下面的BG支持的那四种大小规定的,而是下面我们使用的传递MAP数据的函数决定的。关于这个问题,我想留到下一次《GBA游戏编程菜鸟入门篇4——多层BG及TILE模式下的多层BG》的时候再讲。8 O. C1 C, v0 G& ?! [* k
    & N8 M( v8 z2 C! w, O
    然后写程序了。
    / p1 a8 s2 i! y2 ]) b. k4 ^. ?' y, n2 Z" P8 r$ i0 }1 T$ T
    #include <GsGBA.h>//不用多解释了吧?由于这次用不着显示文字,所以就不用包含<GsText.h>了。 * D5 T; ]5 f; _% T8 N4 s8 ?  D

    8 I- A8 d( y% c6 B#include "BG.map.c"
    % _" S4 G& d; ~2 f) V
    , p5 f5 e0 m6 S3 N1 p! I#include "BG.raw.c" . t! z) U- j/ q
    4 ?  U9 F! l4 j# J4 x* l
    #include "BG.pal.c"//这三个包含,就是把你那张位图的所有数据包含到程序里了。 - T0 p7 A4 h# ^6 M/ Z% ~

    ) k0 I5 |) Y9 E3 C! K% Mint x;int y;//这里定义了两个变量。这两个变量是后面对BG进行卷轴操作的偏移量。
    ! l; M7 F. x- O, y3 Qvoid Vblank();//在这里先声明一下VBLANK中断服务程序7 b) w- A8 {% [' C

    & {) r) Z/ `# K) rconst IntrFuncp IntrTable[14]={
    6 P; \" a; X2 }6 h; u4 u5 X
    1 m) P. L' i: c0 ]0 X+ P  NVblank,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy& `  S( F8 ]/ A5 O* Z7 e
    1 J7 Z1 T* d+ ^+ n2 |) |
    };//注意看这个中断向量表。VBLANK的中断位置,被一个函数Vblank所占用。这个时候,假如CPU响应中断的话,每当VBLANK中断到来,都会去执行一次Vblank函数。
    3 S5 v4 l9 ?1 @1 f  vint AgbMain()//主函数了 : ?& |6 M# L  I9 a$ e: ^2 r
    {
      `. |& L% W! E, W* T6 e
    4 ^, l3 B; t* X1 z7 _% G  GsOpenIRQ();//这是初始化所有中断,并打开开VBlank中断, Key中断,DMA0 ,DMA1 ,DMA2 ,DMA3中断请求。以后如果出现这些中断,CPU就可以响应。
    ( p; O' A! D" k6 i, D; s  GsSetMode(MODE_0);//这是设置模式。这次使用MODE0来显示图片。 # \/ |5 t( b( C4 R" A# s

    ) ?" g3 Z4 `6 E* F4 C( t  GsInitBG();//初始化BG。只要是使用MAP,必须初始化BG。
    & l  T# x5 Y  a0 B0 E3 _6 b
    3 }3 W6 ]) K% B0 N# ?6 E  GsLoadBgPal16(BG_Palette,0);//装载图片的调色盘到GBA的调色盘内存的0号区域。在MODE0下面GBA的调色盘最大为256色。这256色调色盘,又可以分成16组,每组16色。这次使用的,就是这256色调色盘中的第一组,也就是调色盘0号区域。根据前面的表格,在MODE0下面,调色盘可以是1个256色的调色盘,或者是16个16色的调色盘。由于我们的图片是16色的,因此就用那16个16色的调色盘中的第一个就行了。关于调色盘,下次教程我会详细解释!
    2 a6 I1 O$ P/ \! v7 k  GsLoadBgMap(BG_Map,sizeof(BG_Map),31);//装载图片的MAP数据到Char Base第31号区域。GBA图像内存,除去调色盘内存和OAM内存(这个OAM内存你现在暂时不要管它)被划分为4个块,共32个区域,也就是每个块有8个区域。 这每一个块,被称为一个Char Base。 ; r; r- X, H+ u/ l8 q& h2 ~2 O8 f
      GsLoadBgTile(BG_Tiles,sizeof(BG_Tiles),0);//装载图片的TILE数据到第0号Char Base。 ! T  b3 @* a* l3 L& Y
    4 Y. C1 i3 ?& z
      GsSetBg(0,Gs_BG_TEXT_SIZE_256x256,Gs_BG_COLOR16,0,31,0);//设置BG。我们将使用BG0来显示图片。这个BG,是TEXT属性,大小是256*256,颜色是16色,其中MAP数据在BG的31号区域,TILE数据在BG的第0号Char Base,禁止该BG的马赛克特效。 / a' y  W9 X2 T0 _' |3 S6 M
      GsBgPriority(0,0);//设置BG的优先级。BG0优先级最高。也就是,处于所有BG的上层。
    4 x. Y: d( U5 A9 ~
    5 _. o1 Y! V5 K; y% y- |8 T8 a  GsSetBgState(0,ON);//设置BG状态。将BG0打开。由于GBA默认所有BG在初始化之后都是打开状态,所以这一步要不要无所谓。
    + K8 v6 @4 g0 G% |/ r# N* ]
    % i1 |! w" ^' g. T, k, l' g, k  while(1) 6 z: C0 z/ \7 n* ]$ R. N1 Z3 U" w

    3 n" C  n( f1 m8 s2 s3 Y  {  n7 @; t% `, E; s( U2 A0 W
    , Z8 i( p# ^3 ?; R% c: f
      GsWaitSync();//等待同步。 $ T2 g' R* F, _- Q

    " ?7 p4 g5 G( w( {  }
    & E3 H, m. H* p/ \7 s
    9 _: b5 H0 H+ Q8 I  K8 Y1 m$ T- q2 ~  return 0;//返回。
    % W" N, ~# K* d3 @& z! H. T' j! [3 U3 g* `
    }
    + C0 _5 a& ?8 C% W1 m% V; D) ]( m& {' H; C6 Y+ c% e7 j# n
    void Vblank()//VBLANK中断服务。每当中断到来的时候都会执行这个函数。 ) y$ |+ Y+ ]# b) d* I: t. }; i& ]

      F; J  A' t( C2 D. U! ]{
    ( y6 H% J# i+ ~4 U1 H, g9 j8 \1 P
    ; R& {1 @; \. @3 p3 {- o- X# j  KeyRead();//读键盘函数。这个函数会得到你所按下的按键。 2 T; n2 P7 ]; f8 Z1 ?6 B! `

    8 m5 b: g5 D" @  if (Cont & L_KEY)x--;//如果检测到键盘左键连续按下,则变量x自减。
    " N* e: U$ l& `2 T, q
    ) x- A% X, ?5 B3 p3 \: y1 [' v  if (Cont & R_KEY)x++;//不用解释了吧? . ^7 o1 f1 @; z0 r; L: l( u' g

    0 B) C1 F4 I' [: u( X/ K  if (Cont & U_KEY)y-- ;! R2 I7 ]. {( G- |. R6 |8 O$ p

    : B, O  s- s# K& s( _6 U' _  if (Cont & D_KEY)y++ ;! h- c! J0 q6 c. H/ E
    + q$ Y0 W* I$ D1 Z9 F+ U6 y/ q
      GsBgOffset(0,x,y);//对BG进行卷轴操作。 - A3 `, a6 F) T0 u
    ( H3 D. E$ S) k7 S) r
    }/ \. z4 N5 N) g7 x9 ~9 J

    " O: u- k) y( ^) I2 J( \& W行了,程序完成。编译吧。
    - W# r0 e; ?3 |9 J0 @' v. F  e, I0 ]
    2 [% c: U; @0 {8 Y/ c7 |
    - w0 k; G' O$ q! Q
    0 ~1 l& B( C/ g/ t. @) A
    / z, v4 b2 o( n
    - c" ]. n( f% O, |  W$ R% @3 |0 m能够看到,屏幕上出现了你的图片,按下键盘的方向键,图片就开始卷轴。
    $ }, C: p5 K5 ]
    + a( I+ q  F+ J; h8 x5 p% b; c; ~我们来分析一下程序的运行状况:首先GBA打开了中断。从此以后,只要有中断产生,GBA就可以响应了。然后GBA将图片的所有数据装载到内存里。再然后,GBA设置BG0,并使图片在BG0上显示。再然后,程序进入死循环,每循环一次,等待一次同步,而每等待一次同步,图片就显示一次。于是GBA屏幕上就连续不断的显示出了那张图片。然后,假设VBLANK中断发生了。这个时候GBA执行中断服务。首先查询键盘。如果这个时候你没有按键,那么两个变量就没有变化。卷轴偏移为0,也就是不偏移。中断完成之后GBA又回到死循环里,继续显示图片。假如又发生了一次中断,而恰巧这个时候你有按键,于是GBA检测到按键,并得到你按下的按键值。假如是左键,那么x变量自减,于是图片向左偏移一个像素。连续按键,于是图片就连续卷轴……卷轴完成之后,GBA又回到死循环,继续显示图片……
    4 j9 r$ g! E1 J7 A% U- r* y1 h* V! e9 w) T1 {6 ?. G' Z
    OK!这个图片是正常显示了,而我们也可以对它进行键盘操作了。希望你能明白这个程序。下次的教程,我们来看一下多层BG共存;就像上面那张口袋妖怪绿宝石的标题画面一样。掌握了多层BG技术,你可以做出非常漂亮的游戏标题!还有,你有玩过《逆转裁判》吧!对那种图层结构组成感觉如何?下次的教程,我就教你“屏幕上一个漂亮的背景图层+屏幕下方的对话框+对话框里面的文字”,也就是,三个BG同时使用!这三个BG中,一个图层是用来显示256色背景图片的;另一个图层是16色对话框的;还有一个是用来显示文字的……下次的内容:多层BG!理论课内容:TILE模式下的多层BG。另外,下次将要涉及程序调试;这是一个很令人惊讶和振奋的工作,因为你能看到以前屏幕上乱七八糟的突然间就显示出一幅美丽的图片……别忘了准备好你的模拟器们!; ?; J: Q$ B$ s. A7 [' s$ Z

    1 |. _5 n- ^7 X4 @" p2 R5 dGBA游戏编程菜鸟入门篇(四)——多层BG及TILE模式下的多层BG
    / O; q) v# X8 u: a
    - @; D! S! G5 p0 i2 e废话少说,开始今天的教程。为了方便说明多层BG这个概念,顺利操作多层BG,我们这次改变一下学习顺序……
    9 Z, ^# x8 m% B$ @; A0 o6 U" U0 v, @4 o8 |
    首先打开上次你编译好的ROM,就是那个显示一张图片的ROM。  I+ w# _2 |4 m9 W, o

    # ~; v- Y2 T) {4 g) B, A8 R. P# j4 t
    ! V* y6 Y( v2 S! R; ]% I/ c7 y# W

    2 `8 C) l/ N/ m' g# c- A: S
    3 [  s$ z( M  L: G2 z  \然后菜单-Tools-Map Viewer,这个时候出现了一个窗口。( c/ m; q% |! i! F* t/ [
    ' p) R% J) \0 V# @/ \7 d$ R" @
    $ X+ U% S: e1 w% u: o8 ?

    8 k' j3 M2 X) T+ c  Z, D& K0 G
    " N4 A' y1 h1 c3 P1 k这个窗口,可谓调试游戏的巨大帮手之一。我们仔细看一下,这个窗口有什么作用。
    # K2 \: A, l0 j6 R
    ) n7 N7 {: O" M" q* C首先,左上角的“Frame”框架:这个框架是灰化状态,就是说,当前不可用。那么这个框架到底有什么用呢?还记得位图模式吗?还记得“双缓冲”吗?没错!这里直接反映双缓冲!Frame0,就是双缓冲的上层;Frame1,当然就是下层。在我们这个程序里面,使用的是TILE模式,不存在双缓冲,当然这个框架是灰化状态……
    # s# C+ o7 t* E# I
    - J' j0 p2 q, O$ {* v紧挨着它下面的“Background”框架:这就是所谓“BG”了!我们知道,在位图模式下不存在BG的概念,只有TILE模式下有BG。而MODE0下面可用的BG有4层,分别是BG0到BG3。这里就直接反映出了这4层BG。在我们这个程序里面,是使用了BG0用来显示图片,所以BG0有图片。你切换到BG1到BG3看看,都是“空白”吧!( y" u/ O. n" o# w. B( u) |
    3 e% j- h9 [' e4 q: o! N0 C
    在下面两个复选框:分别是拉伸和自动更新。一般,拉伸选项没有很大的作用。不过,那个自动更新,确实有一定作用!在以后的程序里面,可以很方便的跟踪BG。到时候,我们会涉及的……
    ) F  N3 J, B8 s' X5 }) c) B, }. _& S& d% q
    再下面,是一个TILE。当你的鼠标点击右边的BG预览窗口的时候,这里就放大鼠标点到的TILE。一般来说,作用不大。
    ! {- m5 r4 B/ a, Q6 V+ f( c$ D2 X/ U- d5 k3 s; h9 ]; r( B
    最重要的,就是中间那几个数据了!
    & T$ D% F$ I1 F! L* O& L1 N9 a' [- M& n3 [
    MODE:直接反映程序当前使用的MODE。我们这个程序使用MODE0,所以……
    , N8 X5 Y, p5 ?5 }) J8 p$ `1 E  B" r$ W* f6 T2 K6 \
    MAP:这是图片MAP数据存放地址。根据我们这个程序,可以知道MAP数据存放在第31号区域,它的地址就是0x0600f800。关于这个16进制数到底怎么和区域号换算,一会儿你就会很直观地看到了!
    " u2 i: I; M" C$ J' q) l( I8 s2 B+ j- T- H, B7 _6 r3 @- q
    CHAR:毫无疑问了,这是TILE数据的存放地址。我们的程序里面,使用第0块Char Base来存放TILE数据。它的地址是0x06000000。同样,这个16进制数也和Char Base有换算关系。一会儿你也会很直观地看到。' [2 u% H4 z9 t5 R- |, y% x3 E3 p
    7 }$ I0 v5 i3 T! c8 R. n
    SIZE:大小。程序里面设置BG的时候,规定BG是“Gs_BG_TEXT_SIZE_256x256”,所以这里显示256*256。
    - D! q9 m( g) i8 ~  V# @7 p
    ( c; }4 @, c# o: bCOLOR:颜色。16色一点不错!程序里就是这样设置的。4 r* r  z# W( T# t0 U9 s2 g8 L7 m
    & Z4 [+ u. P5 J7 B; ]9 f& U4 H
    PRIORITY:优先级。优先级从0到3依次降低,就是说优先级为0的话,可以遮挡其他优先级的BG。程序里面就是这样设定的。: I0 D4 Q+ B5 b2 h9 D/ y) [

    9 O% X4 @2 u! n- U% UMOSAIC:马赛克。如果你设置BG的时候,允许该层BG使用马赛克特效,那么这里就会出现“1”。我们的程序里面不允许马赛克特效,所以这里显示为0。关于马赛克,我先说两句:马赛克属于BG的一种特效。GBA硬件支持4种特效,分别是马赛克特效、透明特效(ALPHA特效)、窗口特效和FADE特效。完全学会TILE模式之后,你就可以初步控制BG了。这个时候,来点锦上添花的特效那是再好不过的了!到我讲完TILE模式和多层BG的时候,我们就深入特效……
    0 _; S! c0 W; k: S5 u
    1 Z+ e- j0 s6 VOVERFLOW:溢出。这个数据,我到现在还不知道它有什么用……请达人不吝赐教!& {* a) g) f! [3 q- T5 f( r

    2 H" b% X: E% e3 S; r/ k; y再往下三个数据,当你的鼠标点击右边的BG预览窗口的时候,左下方出现TILE的放大图的时候,这里显示这个TILE的详细数据。由于我们有一个更强大的反汇编模拟器,而那个模拟器在这方面的功能实在是强大,所以我一般忽略这里……- G; y: I; a( B* H' Y3 W
    - [( U" ?" l, _+ o  E. i3 B& _
    最后,那个小黑窗口……装饰吧……?
    8 l7 N% s. f1 a& q8 |. C( h9 ~% L- m( |) o8 l# a( b; c# ]1 u9 }
    嗯……差不多了,MapView窗口就这么多东西。然后是Tools-Tile Viewer。" D- C6 Z) b' f: g
    ; k* a( U2 i0 D+ H

    / X; l' U+ g6 O* ^* v* B0 g& P- G4 n# r5 @
    ) l+ }0 W3 K; w3 T. N5 i- w2 g
    1 K, U6 B" f* `: ^
    这是察看TILE数据的天堂。我们继续。
    % H7 G* c  r: h% k% e( Y
    & {8 B+ j& ^8 e* _. d2 Y: D+ a$ VColor框架:我不用解释吧?如果你的图片是16色,那么选中“16”;如果是256色的,就选中“256”。选错了,右边的窗口预览就乱得不像样。
    * S' Y6 Y$ D: D* w9 X1 m7 k; V, [( f  G
    (众人:那么,32768色呢?)雷精灵:哼哼……TILE模式下会出现15位色吗?(众人:谁说TILE模式了?位图的MODE3、MODE5不就支持15位色吗?)雷精灵:位图模式下这个框架就会灰化……
      w+ C; I  t6 d& @* q! w% P  O! D: P
    CHAR BASE:这是最需要关注的!你会看到,这里有五个单选框,实际上,前4个代表了BG的4个Char Base,最后一个是精灵RAM,又称OAM。现在,我们需要重点关注前4个。
    ; o/ Q; D9 l0 R0 A  J
    * [2 o' S( O! Z$ l1 i在我们的程序里面有关TILE数据地函数是这样设置的:GsLoadBgTile(BG_Tiles,sizeof(BG_Tiles),0);GsSetBg(0,Gs_BG_TEXT_SIZE_256x256,Gs_BG_COLOR16,0,31,0);。可以看出,TILE数据被保存在显存第0块Char Bsae。因此,选中最上面的“0x6000000”,右边的预览窗口出现了图片。不过,图片是支离破碎的。至于我们它为什么会这样先不管。: u0 E; l% _9 W- X$ Q/ j
    ! O: v/ V" h* V. f& F7 E+ o' T& K
    现在你修改这两个函数成这样:GsLoadBgTile(BG_Tiles,sizeof(BG_Tiles),1);GsSetBg(0,Gs_BG_TEXT_SIZE_256x256,Gs_BG_COLOR16,1,31,0);。明显,这次是使用第1块来存放TILE数据。这个时候编译出来的ROM,选中第一个0x6000000”时,图片出现在窗口下方,而选中下面一个“0x6004000”,图片像没修改的时候一样出现了。$ u: A; S  t: [  F' Z6 M4 W
    8 q' Q+ T; E/ _/ n% F
    这个时候,你选中“0x6000000”,用鼠标点击一下图片的第一个TILE,中间的“ADDRESS”数据出现了“0x6004000”!看到了吗?这就是第1块的地址啊!+ J4 ]; i$ }/ k9 F& x" i! o

    " @1 P" M$ s! }综上所述,程序使用哪个块存放TILE,你就选中哪个单选框。于是支离破碎的图片就显示在右边的预览窗口中。并且是顶头显示。如果你选中的是别的单选框,要么图片不显示,要么TILE数据上面出现了“空白”……# @& x$ j/ @8 k' v

    & g" i/ B3 b6 ~5 w8 l于是,有人就试着使用第3块了。察看的时候,预览窗口出现了这样的景象。3 Y" Y, L/ U2 R! q8 R& e% }! Y

    , T) z' E9 |7 |% i; l, c: x0 t4 a* ~; o5 k

    ; o) j9 v$ d$ p7 X# W8 q# G. T& i) k
    预览窗口最下面出现了一些奇怪的竖道……这是什么?切换成256色,还是那样。“0x6010000”里面呢?黑的……
    ( u2 Z, d% [% n7 X4 g/ A' O" [& f
    1 c1 l' L- q; @0 \7 D5 V(众人:雷精灵!这是什么?)雷精灵:这是MAP数据!
    4 n: j) b+ f4 _5 P& v' i8 e1 o2 x$ R! X0 i* F2 |
    看看程序,MAP数据是存放在第31号区域的。我前面说过,一个块,可以分成8个区域。那么BG可用的是4个块,分别为0到3块;那么就有32个区域,分别为0到31号区域。所以,在第3块的最低层,那里就是第31个区域,MAP数据就在这里面呢……
    ! p, u" r" f6 _9 `* f" p0 A/ x
    2 n& t) v0 ?$ ~9 XMAP数据的存放位置是受你控制的,因此可以存放在任意一个区域中,只要是不和TILE数据冲突就行!( w2 |$ v0 p6 g0 b2 C
    1 G/ d, N7 [: ^* }( @
    (众人:那么,冲突会有什么效果呢?)雷精灵:试试呀!我们来改一下程序:GsLoadBgMap(BG_Map,sizeof(BG_Map),0);GsLoadBgTile(BG_Tiles,sizeof(BG_Tiles),0);GsSetBg(0,Gs_BG_TEXT_SIZE_256x256,Gs_BG_COLOR16,0,0,0);。这样就是把TILE保存到0块Char Base,把MAP数据保存到第0号区域。编译,看看吧!
    , V6 u0 q8 T( C; h/ Y- D
    & C. X& Y4 x- Q2 n; X看到了吧!图片花了……
    , u- x& c4 `" v. L+ v$ {$ O% s" Z! {# g, A. z3 {
    所以,当要显示图片的时候,千万记住不要让TILE数据和MAP数据冲突!这一条,是实现多层BG的基础。
    " N# a/ y" w8 G/ e
    / R- u% s6 P  @5 f8 @9 ?) j* H好!下一个,调色盘察看窗口。Tools-Palette Viewer。/ M- a5 [! h3 @" F. R6 h6 `

    . X3 c3 T$ `: I9 t8 v0 T* M; z6 w+ D9 T  _
    6 [& Q2 t. f4 r0 k6 B2 V4 s

    9 L4 |! b) l, d左方,明显是察看BG的调色盘的。右方,是察看精灵调色盘的。我们的程序中没有使用精灵,所以精灵调色盘就是空的。5 {7 o- p6 c% @  J; u2 R& h. R$ V

    1 d: k* m: y0 g0 E8 z1 pBG调色盘中,可以看到是16*16方阵,共256个小格。每一格代表一种颜色。可见,这里最多显示256种颜色。如果是位图模式,那么这里就全是空的。
    3 L. ^- Z$ d$ K& }) E1 n! a+ u6 s% @: ^* }0 Y
    这256种颜色,可以分成16组;每一行为一组,分别称为第0号调色盘到第15号调色盘。在MODE0下GBA可以使用16个16色的调色盘,或者1个256色调色盘。这里很重要,请务必记住。
    : r5 n) ?6 X* j# i
    " k6 D, r6 m" K* b1 p$ I- ?那么,回过头来看我们的程序:我们这张图片,是16色的,程序中也设置BG的颜色属性是16色。所以GBA就从调色盘中腾出一组空间来,用于存放图片的调色盘数据。腾出来的是哪组调色盘呢?你设置的:GsLoadBgPal16(BG_Palette,0);——第0组!所以,调色盘察看窗口中,第一行调色盘,也就是第0组调色盘,出现了图片的调色盘数据。) k' a" {) A" O# Z

    : I: U4 @; m" M5 ^, R7 g那么,能不能用别的组?可以!这16组中的任意一组都行,随你的便!你可以改一下程序看看,效果完全不变!唯一变化的,就是调色盘察看窗口中图片调色盘的存放位置……! ]- J* W6 q$ ?4 m4 E' f; {  K
    . f9 J/ m$ d5 i' w: a
    (众人:为什么第0组调色盘似乎没填满的样子?)雷精灵:说是16色,实际上不到16色……你用来转化位图成C语言数组的那个工具——gfx2gba,就是你键入的那一串命令,自动优化了调色盘。没有用到的颜色,就扔了……
    8 k. t+ s# F* t) U1 t2 A8 Q' O4 M4 J8 s% a: x; {
    关于调色盘,我必须说明一点!对于16色的调色盘,你知道的,共有16组。这每一组的第一个颜色,都是“透明色”!也就是说,对于16色调色盘,最左边这一竖排,都是“透明色”。什么是透明色?就是说,这种颜色在GBA上面不会显示出来。你可以看一下察看窗口,第一种颜色是绿色,0x03e0。这种颜色在GBA上面没有被显示出来。我们的程序中,用于显示图片的是BG0,其他几个BG,颜色都是绿色——都被透明了!所以,只显示出来BG0……这个时候,想必你也明白了,在第一个DEMO中,就是显示文字的那个DEMO,我为什么把颜色装载进调色盘数组的第1号而不是第0号……3 z- g1 |* ?  {" L: T- _. e' L

    2 u& A9 S, a+ }% x4 u# m我刚才说,16色的调色盘,第一种颜色是透明色;其实,256色调色盘也是第一色为透明色。只不过,因为256色调色盘可以容纳16个16色的调色盘,所以对于整个调色盘来说,16色的调色盘可以有16个透明色而256色的只能有一个透明色。
    8 L* G- ?6 I- {- W" w) z
    2 Z+ u( a# E' |5 T& X以上所说的非常重要。你可以看看教程3中那张《口袋妖怪》的片头截图:它是采用了3层BG。根据高优先级的BG可以遮挡低优先级的BG这一原则,那么可以知道“口袋妖怪”这层BG优先级最高,以此类推。但是,如果简单的把三层BG叠加起来,那么很明显只能显示出“口袋妖怪”这一层。因为这一层的黑色,把下面的BG遮挡了!所以,可以推断,在这里黑色为透明色!同样,云雾那层BG,黑色也是透明色。至于最后烈空座这层,无法推断谁是透明色……
    3 B1 m3 F+ r# z7 k
    ( x" [; v# w( V$ c; v" @( _, K(众人:那么,我要是自己做了一张图片,想规定某某颜色为透明色,应该怎么做呢?)雷精灵:呵呵呵……有请PhotoShop!
    . C5 I1 q/ f2 d; g5 a2 e& e; a2 L( j* J  ^( F* t- n+ S6 ~7 K
    还是我们上次做例子的那张图片。用PS打开。由于截图太大我不放上去了。# A5 ]4 ^5 r; {& q: S8 A0 j

    9 g4 D' }9 q6 x; n- F然后,菜单-图像-模式。这个时候,后面出现了一级子菜单。. f5 F7 s( z# s) [

    ) y6 V1 z, l# }& h8 @  `  ~( n9 _$ l注意看那个“索引颜色”和“RGB颜色”。如果这个时候默认选中的是“RGB颜色”你就选择“索引颜色”。
    ) p( O& [* I. w" l5 K3 M+ X) g9 o
    弹出一个窗口。看到了吗?这里更明显的显示出了图片颜色数确实不到16色。' i# h. [5 f& Q, B
    ; M; O6 U, B  p1 r: m
    不过我们关心的不是这个。“调板”选择“局部(可**)”中的任意一个,然后颜色数你就可以自己规定了。填上15。记住一定不要填16!否则这张图片无法保存成4位位图!7 G0 @% B" E- K, b' F1 I' v; w% Y9 O  j

    $ b9 k  I. }. d& D然后点“强制”下拉框,选择“自定”。如果本来就是“自定”你就再选一次。
    * `5 F  X5 B! L- b8 S
      Z4 T: Y: D. P5 v/ w5 M9 s这个时候弹出来一个颜色窗口。
    8 j3 y9 x# Q& `6 L
    4 x) G# p3 Y  [( W) C9 W点第一格颜色。一定要点第一格颜色!因为GBA调色盘规定第一种颜色是透明色,而我们正是要选择透明色……
    8 y" J! A. g) t
    # L8 B( A& {% P然后出现“拾色器”窗口。扔在那里不要管它!把鼠标移动到你那张图片上。这个时候,鼠标变成了吸管。' @( V1 c% j8 H& `- z- d- Z
    8 ~. [* Z& N/ ]% Q: ?+ c
    你想让哪种颜色为透明色,就用吸管去吸那种颜色就行了。吸完之后,“拾色器”选定了你选择的颜色。
    1 L0 H9 j$ k" o" f- l
    / w& k# t' K: C& z3 W$ L3 `然后一路“确定”就行了。最后“另存为”4位位图就搞定了。
    % M+ ?/ S, P; w# q* H; s3 R, x; [
    + B( q( ?$ h2 v那么,256色位图呢?一样的方法。自己规定颜色数的时候就随你便了,只要不大于256就可以。填256色行吗?行!
    ( I+ j3 D: J+ G6 f& @% p$ P( w% f, y) T: D2 m3 |- |
    然后就完全一样了。保存的时候,保存为8位位图。- W) E3 l* {. P, Q/ K4 O1 }

    ; l4 }' P9 P2 j5 ^  I$ l/ Y+ h图片搞定了,转化,编译吧!" i% ?3 O: H  F
    1 Y7 ]: L5 y/ V" r$ B9 G  e4 E
    嗯?没变?别胡扯!你看看调色盘!第0号颜色,是不是变成了你选择的那种颜色?
    7 F' z0 x3 I. O" Z, j2 r
    ( i7 U1 y: Y6 j* ]# b(众人:那么,怎么显示出来没有变化啊?)雷精灵:我问你,如果这种颜色成为透明色,那么图片中涂有这种颜色的区域,显示出来会有什么变化?(众人:透明呗!)雷精灵:对!确实是透明!那么,BG0的某块区域透明了,会有什么后果?(众人:如果BG0优先级最高的话,其他的BG会在透明的区域显示出来。)雷精灵:你这不是明白了吗!你看看BG1到BG3。(众人:噢噢噢!原来是这样!)雷精灵:就是这样。如果其他几个BG没有用到的话,它们自动使用透明色。但是由于没有给它们定义调色盘,所以对它们而言,调色盘第0号颜色就不是透明色。所以,它们几个BG,显示出来了。但是对于你用到的BG,该透明的区域确实透明了!只不过,后面的BG颜色是透明色,并且通过透明的区域显示了出来……
    # M4 H" y% W. A; v5 ~3 v' Y! ~0 j2 h0 p- X) @6 R0 q+ E" [* a9 ]/ j
    那么,既然都明白了,我们来做个显示多层BG的DEMO!这一次,我一步一步的写源代码。
    7 I) N5 D6 l2 K, d" Y; A: c8 ]( e+ T% {0 r+ l
    (众人:不是还有几个窗口没讲吗?还有一个OAM Viewer……)雷精灵:我知道。那个OAM Viewer,是用来察看精灵的。但是反汇编模拟器No$GBA在这点做得极为出色,所以你可以抛弃VBA的OAM Viewer。其他几个比如IO Viewer什么的,一般用不到。
    - _9 e: E( I5 I9 Y$ L, x7 w' Z8 _& q  E/ _8 ?+ j1 _
    嗯……首先是图片资源。自己找图吧。
    ! g3 Q' ^2 C4 j
    6 `4 y: Z8 s% `首先我们需要的是一个256色的背景图片,256*256像素。我在这里给它命名为BG.bmp。) |1 N! O3 _6 `2 ~# C
    4 f$ Z. ^  D$ z9 ^/ l# m9 p
    这张背景图片有点特殊要求:还记得MODE0下面的调色盘,最大为256色吧!那么,我们这次的DEMO,是要同时显示这张256色背景图片和另一张16色的对话框图片,还有位于对话框里面的文字。这就出现了一个矛盾:如果将256色图片的调色盘装载到调色盘内存里面,那么,那个16色图片的调色盘,装载到哪里呢?调色盘最大256色,已经没有空间留给16色调色盘了;更何况,还有文字层,至少需要2种颜色(透明色和文字颜色);如果你想要文字描边效果那还得需要一种颜色;一般的RPG里面,为了突出某些重要的剧情物品或者重要对话,有些文字反白显示或者高亮度显示或者红色显示什么的……这又得需要若干颜色。- f: Q. c& G! L0 b7 p
    ; w' O% T& ~4 T: Z* F
    以最坏的情况考虑,对话框图片是16色,那么就需要一组调色盘,文字颜色,也给它分配16色!(15种颜色,显示文字绝对够用了……)那么,那张256色图片,留给它的调色盘的空间,只有224色!
    0 G  W; q; ]) H5 \4 N* b* b' ^) o
    + o* _# a8 a4 ^: _4 ?所以,你找到的这张图片,如果颜色超过224色,请注意要减色!否则,对话框图片的调色盘和文字调色盘将会覆盖你的256色图片调色盘的一部分,导致显示出来的图片出现颜色异常!3 O: ]/ D8 z6 T5 k1 \& Z
    2 w1 S( Y( D( ?* R2 R  w3 z
    减色也很简单。还是有请PhotoShop,还是像选择透明色一样,只不过到了填色的时候,填224就行了。由于是背景图片,也就是它的BG优先级最低了,所以没必要选择透明色。
    " C$ Z& L9 F6 m8 {4 O. x; l$ F, L4 }5 C1 F9 V, A3 d5 ~, x
    6 a2 J) D8 R& x* g8 n: d  _
    6 u: e: _1 ^$ q1 |  [' m3 @& v

    2 ~+ G4 R) ~8 @% T/ F/ n, t& w( s1 k0 O) K8 E
    - p" M! o6 E  X) B2 o
    这张图是我从GBC的《心跳回忆》里面截出来的。由于GBC窗口小,所以做了加工,将它拉大到240*160,就是GBA屏幕大小。然后再拉大到256*256,以适应程序需要。图片截出来的时候,是真彩色(24位色),然后用PS减色,减到224色了。
    9 b/ G; h: @8 {# p$ f; A: y: n# ^# c/ I
    转成C语言数组。由于这次的图片是256色,转成数组的时候,命令有点变化。根据gfx2gba的说明,这次的命令是:gfx2gba -fsrc -t8 -c256 -pBG.pal BG.bmp。
    : z" ^( O: V0 K7 H' T5 ?, O$ Y- L4 E7 t8 e% n5 A
    然后文件夹里面多出了BG.PAL.C、BG.MAP.C和BG.RAW.C这几个文件。- ^* t+ C0 M6 [4 z! C8 a/ }0 b
    4 J/ Z3 v. f# s3 a% w9 O
    然后就是16色的对话框图片了。依然,16色,256*256像素。起名为Frame.bmp。* N9 R) {1 b# l! F' @+ `
    4 w! H% K' i! w! F
    记住这张图片要选择透明色。然后转成数组。
    - A7 j& ?, d- q
    0 ?1 d3 m6 r- t; o% l0 [; B- s9 F  v& X# \$ M# V

    ! L% S9 ^- |' n$ p5 c$ W' P# u0 |7 I$ t; e
    & u* ~/ ~/ V5 B: G
    . s; E  l( Y0 r) j, V& |! K: b, b( \3 f4 R9 C. [# C
    这张图是我从HuangYZ大人写的《心跳回忆GBA》的DEMO里面截出来的。最近我一直在写烧录地带的《心跳回忆》,当时为了能够在游戏对话中显示姓名所以对那张图修改了一下使对话框左上角突出一块。这次直接拿来用了。在这张图片里面黑色被我设置为透明色。
    , y/ y- ~0 g4 b: b* b$ w% S7 U* C( J9 F4 b: p& ~+ ^" v, k
    好,写程序。: Q. y+ a) s1 `5 Y4 Q, d# f

    1 R+ J: i. A! ]( ?) i首先最好规划一下:我打算,让BG0显示256色背景图片,Char Base使用第0块;BG1显示16色对话框图片,CharBase使用第1块;BG2显示文字,Char Base使用第2块;至于Char Base第3块,存放这几个BG的MAP数据……优先级毫无疑问,BG3不使用,BG2最高,BG1次之,BG0最低。
    % ]! Q' Z1 C  W) w  P
    & b' z5 v; h+ X5 C, m; c  F7 C5 c9 ~
    , Q. _4 r! M9 }9 s#include <GsGBA.h>& C5 [, ~8 Z) a3 @
    ( ~8 q1 y* y# x; I/ h
    #include <GsTEXT.h>1 }$ d! R% V, ^' A
    3 c( S7 d) ]  O- v+ v) ?* J
    #include "BG.pal.c"
    ; `# i" {  g7 i$ u) A1 Z, u. L) ~& o' h2 Q/ O
    #include "BG.map.c"' D0 r# F$ z( H$ ~5 l

    + e% p! U* v2 C' U* r7 s( f#include "BG.raw.c"7 ?. F3 ~6 D  Q" S0 a6 W7 D

    & Q9 j: J1 O9 c0 g3 H1 h#include "Frame.pal.c"
    " D1 g, B+ t# h; X) ^' A9 E: v3 G* d, }' \2 }$ h
    #include "Frame.map.c"
    - {9 `  r0 Y( c& V
    # u- e/ B" x: s/ T. F" Q7 s#include "Frame.raw.c"//你也有感觉到,这一堆包含实在是太让人头疼了。确实,一张图片就有三个包含,以后出现的DEMO,万一出现更多的图片,这个包含,岂不是要写很长……所以,我们最好把这三个文件合并。最简单的方法,就是用DOS下面的COPY命令,把三个文件组合到一起。不过,每次处理完一张图片就要打一次COPY命令,也有够麻烦的。最好的方法是写一个小程序(当然这是运行在Windows环境下面的程序),每当生成完那几个数据文件就自动将这三个文件组合。我偷懒了一下,写了个批处理,也自动完成了这样的任务……. u6 G+ R& B# h

    % Q7 W9 O$ P5 @( k. q% m" x2 Lconst IntrFuncp IntrTable[14]={dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,
    % S& b) v5 ?7 i4 y0 o};//中断向量表。这次,仅仅是显示而已,不对它进行操作,不设中断。
    # N0 M' H# F' l( T
    4 P, W( X9 }0 nint AgbMain()# s: E6 C, L/ n! Z5 F6 O
    % Z: I& B9 t8 H+ T8 p5 [
    {7 t/ Y1 a  J3 w& l

    8 j- H# c$ a4 Y9 z7 P+ n1 jGsSetMode(MODE_0); $ H! y  a8 |, P% j$ o. G
    : \1 `2 G% h+ }' {( W3 d4 F" d$ K
    GsInitBG(); , R5 O/ }; {+ D" F+ \6 [. J

    - `; R  d. i+ C' {7 p6 UGsLoadBgPal(BG_Palette,224);//载入256色调色盘。这个函数的参数请参阅帮助文件。我前面说过,最坏的情况下载入256色调色盘时最多载入224色,所以这里调色盘载入字节数设为224。
    % {2 M2 g( [, d0 l  B: j8 e: Y, ^9 t9 Z& H2 i7 L
    GsLoadBgMap(BG_Map,sizeof(BG_Map),31);//还是一样,将256色背景图片的MAP数据装载到第31号区域。
    ' n5 @3 F( Q2 K5 y6 W
    2 B# g- S+ W. {GsLoadBgTile(BG_Tiles,sizeof(BG_Tiles),0);//还是一样,将256色背景图片的TILE数据装载到Char Base第0块。
      S4 H6 Q. T; f% q
    ) E  B. n: L1 v" H, z5 A( `; w/ {GsSetBg(0,Gs_BG_TEXT_SIZE_256x256,Gs_BG_COLOR256,0,31,0);//设置BG0:TEXT属性,大小是256*256,颜色是256色,其中MAP数据在BG的31号区域,TILE数据在BG的第0块,禁止该BG的马赛克特效。
    & i+ N4 Q& Z( g" g, G! x; {7 X
    1 I" o; t  V3 S/ u, C1 qGsBgPriority(0,3);//设置BG0的优先级。最低。$ F( m. _; N/ z# i% D+ O
    0 {: l# [; g$ v! C% E2 u' {: p: }
    GsSetBgState(0,ON);//打开BG0。写到这里,如果就此停止,再写上死循环、等待同步和函数返回,那么编译之后,就应该显示出那个256色的背景了……不过我们可不能就此停止,后面还有两个BG呢……
    4 x1 h4 Q3 H% L2 o! j* q
    , @/ N4 w6 H/ @* X* N7 M2 d/ z; TGsLoadBgPalEx(Frame_Palette,224,16);//这个函数是实现多层BG调色盘的关键。假如,我直接载入BG1的调色盘,那么它将默认从调色盘0号位置开始存放。这样做的后果就是将BG0的调色盘覆盖了一部分!导致BG0那张256色的图片颜色不正常。所以这里必须要用这个函数!根据帮助文件,可以得知Frame图片的调色盘将从调色盘内存第224号开始存放,存放16种颜色。这正好和上面那张256色的调色盘相接起来,既不浪费空间又不导致冲突……; ]% I1 c- S1 Q9 G8 ?, s) A1 l. y
    $ G' Y! ~8 Z* F. u6 A0 E
    GsLoadBgMapSelectPal(Frame_Map,sizeof(Frame_Map),30,14);//这个函数是实现多层BG MAP的关键。既然Frame图片的调色盘是从第224号位置开始存放的,那么对于MAP数据,也应该从第224号位置(也就是第14组调色盘)读取调色盘……(众人:MAP数据与调色盘数据有什么关系啊?)雷精灵:怪我怪我,我当时没有说清楚,MAP数据里面保存TILE数据的坐标,还有一些调色盘的索引数据,TILE的形状和翻转数据什么的……这个不重要,所以我当时没讲……7 G) w) M# _" r7 c. @

    $ {0 e, x8 _" u8 o" ~) [  GGsLoadBgTile(Frame_Tiles,sizeof(Frame_Tiles),1);//这个不需要变动函数,直接将16色图片的TILE数据装载到Char Base第1块就行了……
    5 X$ X* j2 P/ I; ~
    : ]% U( C1 I. R: P6 ?  JGsSetBg(1,Gs_BG_TEXT_SIZE_256x256,Gs_BG_COLOR16,1,30,0);//设置BG1:TEXT属性,大小是256*256,颜色是16色,其中MAP数据在BG的30号区域,TILE数据在BG的第1块,禁止该BG的马赛克特效。
    $ G9 I) y( O2 [' K6 o$ W8 H4 [% x9 T  s0 _
    GsBgPriority(1,2);//设置BG1的优先级。次低。
    7 f* L& K! Q- a3 {0 a1 B; _. A" c5 U$ ]4 Q- x1 \* s% x
    GsSetBgState(1,ON);//打开BG1。接着是文字层了……  |" g/ s! H& t
    ' k2 {' m8 w: C  G* s
    BGPaletteMem[241]=0x00ff;
    $ X( h' @1 Q& F# ^0 n* `3 j0 {
    GsSetBg(2,Gs_BG_TEXT_SIZE_256x256,Gs_BG_COLOR16,2,29,0);//设置BG2:TEXT属性,大小是256*256,颜色是16色,其中MAP数据在BG的29号区域,TILE数据在BG的第2块,禁止该BG的马赛克特效。
    - ?: l6 |) b0 Q+ D& E4 p
    9 F' i- u4 W) N; ^1 S. _GsTileTextInit(2,2,29,15);//初始化文字层。根据帮助文件,被初始化的文字层是BG2,MAP数据将被保存到第29号区域,使用第15组调色盘。而那个强烈推荐为3的参数,被我改了!我为什么要改呢?你想一下:这个参数是CHAR BASE号,既然文字层使用的CHAR BASE是第2块,那么这个参数当然不能再用3,而只能用2了!不信?你改成3看看!
    & a9 u+ O( W: t9 ]
    8 r- c  y/ n/ R, [; Q3 U# U8 iGsTileTextOut(20,110,"我爱TGB,我爱烧录地带!",241);//……% U- A8 i& m% V: \, N
    7 \6 K9 N& k% {& q* v
    GsBgPriority(2,0);//设置BG2的优先级。最高。
    0 C2 E+ Y: D3 e. c7 F4 X
    8 |+ j! Q5 k$ o3 {! P& n+ P5 ]1 hGsSetBgState(2,ON);//打开BG2。
    8 |1 k4 c1 }( E1 s8 W6 u4 s" i* o' {3 f1 _) ?  e, W; @
    GsSetBgState(3,OFF);//关闭BG3。没用到的图层,最好把它关了!
    8 a, |9 i4 H8 t$ Y# S0 R  q/ v! H. g; f* V& v; H8 @
    while(1) 0 B3 Y! h# }3 |/ N' l/ w
    {' z- e, C+ {$ B
    GsWaitSync();
    ( `% Q: J9 a1 ?}//死循环里依然是等待同步。% `% K7 m+ m5 M& Y; B5 \
    return 0;
    5 I; K6 h% {7 B, O}
    : ]" t& @$ S' m% B; b1 o9 e" I  r$ a9 t: h# r' N
    程序结束。虽然看起来很多但是分成块之后就一点都不难了!编译看看结果吧!/ `0 }9 s# D9 T: h) x  u4 J

    8 X' @( P3 P6 y4 @9 P$ G7 P* O) {1 z0 x) S4 O7 n% g. V) r, `

    * S, f( R+ q1 S9 B) o. Y5 c: i& f$ ~4 L6 g' l
      P5 g! ?/ h3 B+ b

    ( O# e0 s+ S/ ~" V; X9 r
    8 o, k6 K8 q& j/ @$ q# `; Z/ H" a* `4 N, t6 n+ [
    表面上看,确实相当不错!美中不足的是,那一行文字,有点偏上!最好能把它拉下来,放到对话框里面!这个问题怎么解决呢?我先不说答案,大伙讨论讨论,应该怎么做?
    9 Q% k) N/ _+ T3 ~  ^0 f% L3 q+ U6 Q+ `* |0 h' o
    (one of 众人:简单之极!文字函数中,不是有控制显示坐标的变量吗?例子中文字的y坐标是110,那么,增加y坐标,文字不就降下来了么!others:嗯……嗯……说的有道理!)雷精灵:呵呵……我不信!你试试!
    / e- s: N8 [5 j* e( C' A2 q1 }
    ; n# D7 K0 X5 R+ c. k* e4 X5 F(the one who just expressed his ideas:嗯?怎么会这样?文字被砍掉一半!others:是呀是呀!我稍微降得多了一些,文字就全没了!)雷精灵:再仔细看看文字显示函数!
    $ h* s- t" J$ B  E# n% s
    4 l: I- ?' @; J! L+ q只见帮助文件上赫然写着:“参数:int x,y : 坐标 注意,y必须在0~120之内”!: v' E  T3 p# T
    9 q. R) O9 c! B# R+ J
    (众人:那怎么办?文字拉不下去啊?)雷精灵:呵呵……只拉文字,确实是拉不下去……但是……
    9 @$ T9 T, B9 H; i. m" b+ F9 i+ N$ [( q1 a- b
    (another one:……但是……如果连整个BG都一起往下拉的话……)雷精灵:对!就是这样!对文字层进行卷轴!& n- B8 W, ~% a: C

    / ^# k& c! z! o+ `将这个函数添加到显示文字之前:GsBgOffset(2,0,-10);然后看效果……* ~; P' D; J. u

    # {# N5 y; u3 F
    $ w' o: I, F/ ~3 y/ Q/ {8 V  g+ i+ k/ z, r+ q& q7 w  n$ b; F

    7 h% r' g6 F1 T# z$ K) F( M好,这次的DEMO就差不多完成了。如果你对上面的代码还是不甚了解的话,建议你再从教程2重新阅读一下。如果你彻底明白了,那么,我们来玩一个游戏:找不同!你看DEMO中背景图片,和上面那张源背景图片有没有什么不一样的地方……找到的话,你就看一下TILE VIEWER。注意要仔细看看CHAR BASE!如果你发现了什么,那么,你就有能力继续看下一篇教程了——《GBA游戏编程菜鸟入门篇5——Char Base冲突》……下一次的教程,我们来彻底根除这次的DEMO中出现的问题。理论课内容也是Char Base冲突。
    " x; Z/ A4 g: ?3 h' d: I

    , X1 Z3 O* R7 Q" C* d- ?8 j* G8 i

    ; c# _# _/ Y( B9 @. b6 F
  • TA的每日心情
    别闹
    2013-12-11 16:57
  • 签到天数: 129 天

    [LV.7]王国居民III

     楼主| 发表于 2011-9-13 09:23:29 | 显示全部楼层
    GBA游戏编程菜鸟入门篇(五)——Char Base冲突 % w  H4 v' A6 t/ q- |

    5 Q. n/ G2 {% N/ T" v+ i( N  x上次的教程实在太长,我的手,打字都快打抽筋了……帖到论坛上的时候甚至发生了“发帖内容不得超过15K”这样的警告!我不得不把教程分成两次帖上去。但是这同时引起了代码错误,所有颜色都丢失了。这是没办法的事情,唯一解决的途径,就是让管理员将论坛限制放宽一些……扯远了……这次的教程很简单,内容也少,不过,不是很容易理解……大家做好心理准备!
    % }# E8 `, k& Y* v1 }6 h
    , \2 |. T1 U+ f# n- p想必聪明眼尖的朋友一定已经发现了,上次生成的那个DEMO中,256色背景图片,楼房窗口处和源图片不同。察看TILE VIEWER,发现原本用来存放Frame对话框图片的CHAR BASE,出现了其他数据!; D5 X' J' C) y
    % t; a8 j! y0 @. V
    + G0 Z; M+ I5 r) C. A+ E
    / Q$ s: j! m# m5 U

    9 ?% K- N2 m$ X# L' J+ r3 G3 T' p! ?/ ]8 e8 J0 m( N9 H

    9 x7 T  u3 S% V; F用16色察看,只能依稀辨别出来,最前面几个TILE还有点像Frame的某些TILE,后面一堆TILE,就不知道是什么了……用256色察看,竟然发现这是256色背景图片的下半部分!4 a1 l1 w4 f+ S
    3 ~$ L2 w' w# I! J
    这是怎么回事?! V/ G9 V" p6 M# G
    9 m2 t$ h1 U: Y8 x
    回过头看看程序:背景图片是存放在0号CHAR BASE,Frame存放在1号Char Base。存放在0号的256色的背景,怎么会跑到1号里去?% e8 o; \! P3 s. a# I
    * a/ t; r; }  {+ x' k) O" n1 }
    (众人:0号Char Base不够存放背景图片,导致溢出的部分跑到了1号里面去啦?)
    " c8 W" G) ?# X# M+ e& ^; J: R  G3 r! `% [9 \- G. ~
    雷精灵:强啊!你怎么知道的?/ O8 C% a5 E: @  Y& W  A. W3 o! i5 l

    : u. w; E0 y; c7 g: q' K6 h(众人:把两个图片的TILE文件打开,明显BG的TILE远大于Frame嘛!而每块Char Base的空间只有0x6004000-0x6000000=0x4000这么大,万一BG的TILE大于Char Base的容量,那很明显多出来的数据会顺序存到下面的Char Base里面去啊!)
    , Y4 ]2 i7 S# ]# B: a+ z0 z' Y' f; y3 m8 I
    雷精灵:Exactly!那么,怎么解决这个问题?% t: s) o3 g% ]! @# ]% |4 H- z
    1 t& `7 G  |; P0 v2 J
    (众人:……)" g; \% v+ A% f! g2 `3 I: f

    * m+ b( w: R$ ~1 [2 ]* J- c) x雷精灵:(得意地笑)最简单的方法,就是不用1号Char Base存放Frame图片了。根据程序,0号已经用来存放BG图片,1号被占用了一部分不能使用,2号存放文字,那只能用3号!7 K/ s7 R" W8 O$ ^7 p

    1 e# \5 j, ]& V(众人:慢着!3号里面有这3个图层的MAP数据!你也不能用!)/ x; g6 x( P7 t: I

    9 n# G/ j. X  V" _. A9 ?( r雷精灵:没错。但是MAP数据很小,我让MAP数据搬家,腾出空间来存放Frame图片的TILE。
      X4 z( B5 w4 I( k* j* T2 K) n* ]+ c2 O  y9 q- k
    (众人:那么,MAP朝哪里存放?)1 Q) L0 U4 \# p1 g. Z1 T
    + B) K; k5 v& y- S$ L$ j
    雷精灵:1号Char Base不是被占用了一部分么?把那没被占用的部分拿出来存放Map数据吧!
    9 @; V& o* U7 ^! L$ z  y
    2 ?, q" S& @: u: ]6 F" E7 M* \(众人:(怀疑的目光)行么?)7 Q* H" g- k* z- N$ o3 K

    2 T$ l; a2 B( Q雷精灵:试试吧!
    : A& c  |% x) @, r! j5 Z6 f- Q9 x  z7 @  K
    从主函数开始吧!: S3 V% Y5 E( `0 A; ~+ a% r5 y8 s: s

      k) Y( y6 z1 q: K" m& Uint AgbMain()
    ( N  ?5 j: }) [/ L( t( I# d+ `8 }2 [0 I( q: G' z4 D
    {
    . h: y1 ]* T1 q9 z
    ) h  M; O" l9 J3 X& yGsSetMode(MODE_0);
    : L% W0 `1 ?! c0 c* X( V( A: H: x; X( g$ Q  _, Z- W4 T0 A1 i
    GsInitBG(); / I" D8 i& E# {
    * P  B; j! q1 d. c
    GsLoadBgPal(BG_Palette,224);//与调色盘无关,不用管它。 / K. Y' r% @2 ~0 }9 e; {
    ! D8 _- V, q' X0 ^7 R
    GsLoadBgMap(BG_Map,sizeof(BG_Map),15);//这次将256色背景图片的MAP数据装载到第15号区域。 为什么是第15号?你想……一块CharBase容量为0x4000;一块CharBase有8个MAP区域,那么,每个MAP区域容量就有0x4000/8= 0x0800。你看看在1号CharBase中,哪块区域还没有被占用?(0x6007800-0x6000000)/0x0800=0xf=15!那个数据0x6007800怎么得到的?打开Tile Viewer,选中256色,切换到1号Char Base,看到图片下面的“空白”了吧!鼠标点上空白区域的第一个Tile,看看它的地址……
    7 T6 y! @- D1 n' Z& GGsLoadBgTile(BG_Tiles,sizeof(BG_Tiles),0);//还是一样,将256色背景图片的TILE数据装载到Char Base第0块。 & w/ K) G7 e; a( o' R/ D2 C

    / I7 x* K( W: s2 r/ ]4 iGsSetBg(0,Gs_BG_TEXT_SIZE_256x256,Gs_BG_COLOR256,0,15,0);//设置BG0:TEXT属性,大小是256*256,颜色是256色,其中MAP数据在BG的15号区域,TILE数据在BG的第0块,禁止该BG的马赛克特效。
    # i2 e) O" O& U
    - Q1 [" k6 [0 C* E2 @5 ]GsBgPriority(0,3);6 [8 Q" B2 j- ]% S' _
    % L4 v2 V/ L1 T
    GsSetBgState(0,ON);# g! L3 a" |$ H6 t' H
    & R" j$ Q6 @% ~" B3 w9 o
    GsLoadBgPalEx(Frame_Palette,224,16);* j; d- I1 P- E% q) E5 j

    + d: k$ ^. i+ SGsLoadBgMapSelectPal(Frame_Map,sizeof(Frame_Map),16,14);//那这里,明显就存放在16号得了!$ {# r$ W% ^" ~) a/ d; Q; H2 o; D

    3 R) F! w4 I# D. l2 |& x) E4 GGsLoadBgTile(Frame_Tiles,sizeof(Frame_Tiles),3);//这个不需要变动函数,直接将16色图片的TILE数据装载到Char Base第3块就行了……/ f1 D9 Q1 E3 A/ R/ A
    GsSetBg(1,Gs_BG_TEXT_SIZE_256x256,Gs_BG_COLOR16,3,16,0);//设置BG1:TEXT属性,大小是256*256,颜色是16色,其中MAP数据在BG的16号区域,TILE数据在BG的第1块,禁止该BG的马赛克特效。7 `. O8 d6 b1 a9 T7 n- X1 P
    3 Q' a: g4 q- s$ D* t
    GsBgPriority(1,2);//设置BG1的优先级。次低。
    1 k8 K% ^' ^7 M9 S" Z4 U6 @- W- ^; B% X" u( P" L# J
    GsSetBgState(1,ON);//打开BG1。接着是文字层了……$ J4 O1 a" [( D
    ; \+ \- R. \- N: K! o- P
    BGPaletteMem[241]=0x00ff;8 O, t+ t2 x! K! f6 O

    % y5 S3 l3 Q+ `1 F. o- t/ jGsSetBg(2,Gs_BG_TEXT_SIZE_256x256,Gs_BG_COLOR16,2,17,0);//设置BG2:TEXT属性,大小是256*256,颜色是16色,其中MAP数据在BG的17号区域,TILE数据在BG的第2块,禁止该BG的马赛克特效。7 P7 Z1 {1 B% A& f) y
    5 }* ^; F. N4 f4 w' ]
    GsTileTextInit(2,2,17,15);//那这里也是一样了。. ]8 w$ R7 P! H

    : `+ x+ N% _6 r' iGsTileTextOut(20,110,"我爱TGB,我爱烧录地带!",241);//……
    3 U0 R/ f) a* z9 s3 \2 @& k9 f4 z
    & \. H# J1 @- f8 P- `  E+ T0 I/ NGsBgPriority(2,0);//设置BG2的优先级。最高。
    2 C4 n' S$ J/ J7 Z# }
    # w2 J. Y5 {0 t0 H5 tGsSetBgState(2,ON);//打开BG2。
    * t$ v7 r8 u  Z1 {7 N- g
    * {, r1 `0 X& E  M- GGsSetBgState(3,OFF);//关闭BG3。没用到的图层,最好把它关了!, @* ]7 L  f8 T/ ]

    0 I& z# @# d) E( Y4 s7 \修改完了,试试看吧!; p9 @2 G& U* R
    9 x) e! W- H  f1 m7 \0 m
    3 e5 g/ C; ?: @! x! n* I' t

    4 j; B$ x3 p. ^3 n( e  i4 @1 c+ K6 O$ K' D

    ! ^9 Q# @# y+ W* u什么!!!
    3 }: H$ a2 A" W+ p* B
    0 k: F( m( Z: W1 S哎呀呀……疏忽了!看来1号CharBase没剩下多少空间,把BG0的MAP载入的时候,还可以容纳,但是这个时候已经满了,BG1和BG1的MAP,跑到2号CharBase里面去了……
    5 a: U- x* }1 P) i0 T# `1 n
    - D7 q8 }+ \* X没戏了,MAP还是放回31、30和29号区域吧!好心没得好报!, j; E! K. K! f* U; V

    0 f, S5 ]/ c# s( P9 H- n) Y% M) H6 k; C$ B6 e# t! n5 N' A

    : q1 R: B& d/ ^1 H; H
    ' e. q- N* J" K6 i8 N
    " c" {. Z6 \1 k1 x& A+ \终于行了……有够累!# Z/ s+ D5 E  U( |

    ! u2 X3 X1 Q1 U0 l- B0 C3 l3 y从这个DEMO,我们能看出来,在作多层BG操作的时候,请多加留意!对于比较大的图片,你必须充分考虑到这图片的TILE是不是会超出一个CharBase所能容纳的范围!如果超过了,那么它的下一个CharBase就会被占用一部分,从而导致这个CharBase不能再容纳其他BG的TILE数据……当然,可以将其占用之后剩下来的空间作为MAP数据的存放空间,但是,你仍然需要注意剩下来的空间是不是足够放MAP数据……对于2层BG来说,基本上不会遇到这种情况,即使这两个BG都是256色。不可质疑这仅仅是“基本上”,无论如何你需要留心!相比之下3层BG,尤其是其中有一层BG是256色,这种情况出现的几率就会大大增加……如果是4层BG,仅仅是“如果”,那么这种情况出现的几率就是100%。这个时候请务必精打细算,仔细寻找可以容纳MAP数据,而又不会造成冲突的空间。你必须耐心的计算,到底哪里可以容纳MAP数据,到底哪里可以存放TILE。实在必要的时候,穷举法也不失为一种方法……
    * ~- l1 \) a/ ?% v7 N) ^: M. J; G" {3 r3 W8 F# ^/ m) S8 Z
    但是,通过最近自己写的DEMO,我发现,某些图片,如果比较精细,颜色数较多,那么有一定几率会出现:这张图片的TILE数据甚至2个CharBase都不能容纳!这个时候千万不要进行大于2层BG的多层操作!否则,你会很沮丧的发现根本没地方容纳另一层BG的数据!这个时候你面临的就是,牺牲图片精细度和牺牲表现效果这两个抉择……不用我说,这两个抉择都是很痛苦的……  _; h7 {" N/ l& L2 e
    , y" q  z6 Q7 X1 E' Y
    嗯……差不多就这些了。到现在为止,如果你能完全理解上面几篇教程中的思想,那么你就掌握了GBA编程的三分之一。诸位不要说我罗嗦,再次强调一遍,TILE模式下的BG技术很重要……想进入GBA编程,请一定要掌握这些内容!" q, a7 l& ?2 z2 [4 L0 x3 X

    6 i+ a- Z9 Z2 Q那么,难点已经过去了,下一次我们来点轻松的——特效!在特效的配合下,BG就会变得多姿多彩!关于特效的调试,基本上不需要。你只需要通过VBA,看看效果是否符合你的想法就可以了。另外,我推荐的两种模拟器,在表现特效方面有一些明显的不同。具体地说,No$GBA表现马赛克特效的时候,是将这种特效作为水波特效处理的;表现FADE特效的时候BG和精灵不一样。由于在GBA机器上,特效的效果是和VBA一样的,而且VBA远比No$GBA普及,所以你就以VBA为准吧。下一次的内容:硬件特效!理论课内容:GBA的BG特效。! ~; \0 Y# r; \% R
    3 O( l( K: ^: {
    . ?# d, O3 L6 e9 }0 J! y' V1 f8 D
    0 o5 @* W" [0 h; p( n! {" P( |$ h

    + S# @( G' x! K; S3 e! }7 K
    2 S. a9 Y& m; M; J( I
    & c6 S  y0 U$ Y+ H7 n$ L1 q7 O. J& s4 Z2 `; h/ U0 c
    GBA游戏编程菜鸟入门篇(六)——GBA的BG特效及硬件特效
    " o3 K# y- @3 J2 u
    : O8 d, q+ ~$ e& \) _1 o特效,不是BG的专利,精灵也有特效。不过,我们暂时不讨论精灵特效。到了《GBA游戏编程中级进阶篇》的时候,我会详细讲解精灵,当然精灵特效也会讲解……& E4 l- o* M0 _0 }2 L+ ~+ W

    / {3 q. s8 C, r6 b对于GBA来说,硬件直接支持的特效有4种:马赛克(Mosaic)特效、透明特效(Alpha)、窗口特效(Window)和明暗特效(Fade)。而通过相关算法,又能实现若干软件特效例如烟花特效、水波特效、颜色滚动特效之类的。其中有一个比较出名,就是伪3D特效。GBA超级大作、人世间最强大的GBA游戏、不到长城非好汉不玩这游戏就白活的经典、人类智慧最美丽的结晶、(后面省略形容词10000字)的游戏《黄金太阳》,将这种特效发挥的淋漓尽致、入木三分、(后面省略副词10000个)……1 ^7 I9 U' s! V; O" N* Q

    $ x; [0 D5 V* s这一次,我们先看看BG的硬件特效。至于软件特效,因为这要涉及对TILE和MAP甚至调色盘的数据操作,而这种数据操作还不是一般的位操作或者循环就能完成的,所以我打算在《GBA游戏编程高级技术篇》中讲述……/ @0 v% a; {# Q" Z
    , C7 M  K" R$ y, b+ d+ [2 h0 C
    首先是马赛克特效。这种特效在现在的GBA游戏里面很常见。例如《口袋妖怪》,当口袋妖怪中毒之后,主角在大地图上行走的时候,每走4步画面就马赛克一下然后迅速恢复正常同时伴随着中毒掉血的效果音。再例如《恶魔城-晓月圆舞曲》中,游戏刚开始的时候主角来须苍真醒来,发现弥那小姐坐在他身边,阿鲁卡多站在右边。苍真醒来的过程,画面就是慢慢由模糊变成清晰的马赛克过程……还有几个GBA的Hgame,里面的马赛克特效实在是令人不爽……(众人:这小子有够无耻……)6 V( J8 V) X% O8 K; T4 i( X
    9 E% _0 U( P4 m  K) L& r
    马赛克特效的原理是什么呢?这不是很重要,但是了解一下也不坏。假如GBA硬件不支持马赛克特效,你知道它的原理后也能自己写出相应的算法,实现软件马赛克特效。
    9 M+ m/ Y' P% J/ A( N! ]& S8 w8 w# Y6 e* _( v" p
    规定图片的左上角坐标为(0,0),向右横坐标增加,向下纵坐标增加。假如要做双轴3倍的马赛克特效,那么,横坐标和纵坐标均为3的倍数的坐标点例如(0,0)、(3,0)、(6,3)、(6,6)之类的坐标点,其左方和下方两个坐标点处的像素,均变成横坐标和纵坐标均为3的倍数的像素。我画个图表示一下:9 @! c+ y; T& t) z7 Z: k4 k* G

    : s6 z9 ~; f+ k4 E+ I6 f! Q; i8 r: ~3 d

    , ~9 V# p* z$ H9 |9 r! P3 k: m+ _$ g6 p
    8 a" {3 }- h9 x6 R( p% z$ @$ v
    . u# _8 R4 w/ T9 b5 C& s8 D' U
    涂有同一种颜色的像素,均变成它的右上角那个点的像素。这样,图片就形成了一块一块的明显的方块。
    ( D1 c  f7 i8 n" t3 |) M- n0 P
    & l7 C+ F% E, b7 i% {那么,同样的,要做x轴4倍y轴5倍的马赛克特效,就是横坐标为4的倍数,纵坐标为5的倍数的点,像素值变成它的右上角那一点的像素值……可见,控制马赛克化面积的变量就是那个倍数。在GBA里面,横坐标和纵坐标的倍数都可以由你控制。
    2 e, n. Z* q; t; E2 G
    * |) Z% w# d% I4 `那我们就做个DEMO看看效果吧!' A* [) M$ x% z" c( g( R2 B& s0 J

    : o, n4 c3 f) ^3 p: |& g% M/ e  N& }- x就拿写单层BG的那个例子吧。我先照抄程序,到需要修改的地方,我会加上注释。
      n' m. e# {4 z  g
    ' W( b6 q; ~) z4 c) w#include <GsGBA.h> - O( J+ s7 s# u6 }
    5 f6 N7 W2 j9 E  P' z7 _/ u3 E
    #include "BG.map.c"
    7 Z! b9 \. G2 |" b7 K; R4 S$ A
    1 B. y7 r$ l% `* Q) U! n#include "BG.raw.c"
    * c0 r! q" ^: H9 u  d9 ]+ K2 |5 i& I' b4 I: ?
    #include "BG.pal.c"
    3 S- t2 S! @* V1 B& a. z$ w# ?; a( i( K& Q4 k
    int x;int y;//这里定义了两个变量。这两个变量是后面对BG进行卷轴操作的偏移量。) m) x  L. L5 G1 }. j/ \

    ! n* I0 y' p& L5 g) y. W- Z4 F+ |int mos=0;//这个变量是用于马赛克特效渐变的变量。
    3 p/ G# w: Q! k! ^, O4 b) J; }8 h$ U; G; u
    void Vblank();
    " b' o. \) u: j4 C0 O- K% ]; I% f
    " l0 Y4 |* H) ^2 a5 }const IntrFuncp IntrTable[14]={; _/ g: ~+ o$ e. Y
    # H9 w# e& K* L( s( p9 D4 @
    Vblank,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy' E% A' K$ c, Z" j, Q

    + |) \. ~* G- t) l' G* F3 [: |};; Z# e/ Q: g! e1 @  y' T" g" _

    9 ~6 W( V. {$ c1 p  K% T! jint AgbMain()
    0 f. y* y7 o' u4 P4 @1 N6 {{! W& v4 \- O( N3 V8 [
    ) Y" f2 n& _' ^8 K' _
    GsOpenIRQ();
    * ?# s2 @; p- L& N! L- j( [( ?! S# Y+ }! i& n0 Q
    GsSetMode(MODE_0);
    ) l$ p5 d4 c, }6 O1 E4 o( O' J. U8 ^" [* `& D
    GsInitBG(); 2 z2 v0 g0 R1 L4 D8 N; ?0 A3 D

    / B; s, J  N3 L" d: I9 N" c- JGsLoadBgPal16(BG_Palette,0);( h4 Z, d- p6 K, K
    6 B. M/ S- A& I' ^0 `
    GsLoadBgMap(BG_Map,sizeof(BG_Map),31);
    # h# d& D) ?' _7 e, Y+ |
    * w5 c' ~! e4 c" xGsLoadBgTile(BG_Tiles,sizeof(BG_Tiles),0);1 w% v" J" q' P/ J5 u

    8 o: b/ }  G9 z8 d6 y8 b2 R; a5 ?GsSetBg(0,Gs_BG_TEXT_SIZE_256x256,Gs_BG_COLOR16,0,31,1);//最后一个参数是控制马赛克特效的,打开它。6 x4 J5 g% ]+ y5 _; Z# ]
    & Y3 N; v5 e" ]7 p+ R. h
    GsSetEffect(BLD_BG0_1ST,BLD_BG0_2ND);//对于硬件特效,设置完BG之后一定要记住打开特效支持!否则后面无论你怎么写,都是没有效果的。这次只针对BG0做特效,因此这个函数的两个参数均选择BG0。如果有两层或者两层以上BG做特效,相应的参数就要小修改一下。1 ]2 t$ Y. n) y+ D2 ~

    . A, Q8 P, A5 d* i/ b! h& f) Q, NGsBgPriority(0,0); . C2 @+ d, ~2 }4 V! H8 I6 w

    ) C9 X$ U/ ?0 I* {: ~GsSetBgState(0,ON); 2 m& L# h5 c: W

    : p, R/ H. ^8 b3 M/ N2 r$ ?7 M  C# nwhile(1)
    3 {# A8 m, s) t# ?9 `( f
    + A- k5 r1 C6 f+ v9 s0 u{4 o! e3 r& v/ ?5 r( l
    " k0 t4 V8 Y: k
    GsWaitSync();
    - C4 g8 ?* B0 @- V+ r) D5 a1 X/ \$ J. m" g7 P+ Z) C  ^7 K
    }
    / z) N8 Y/ X4 V! i& `& N+ G9 ^9 D3 J1 @; z5 o
    return 0;
    2 q) \" v( r, n0 [# |! H- g. o4 T, R
    }, Y2 z7 X0 h/ Q6 j0 E) [+ w" D

    5 ^- u5 U; H- f) ~/ m" h9 g- Hvoid Vblank()
    1 h' {/ O' l! ^4 \6 D
    0 p9 I; @0 w1 _/ d: t) ^/ G8 D{( G( m+ P2 m: D
    : _9 ?  E6 {3 [8 A) d7 x  e
    KeyRead(); ! f- o1 J  N! |# U( }7 t

    8 k6 k- L) V2 g  R: r& G* h, V3 Lif (Cont & L_KEY)x--;
    # X: S& u" L8 A8 j6 _  T# U/ S5 S/ q, L6 Z3 |! o  Q4 [# k* j& p3 K
    if (Cont & R_KEY)x++;
    / Q0 Z7 k, f; E6 W4 j1 e# R, U0 [
    ) c5 I3 i0 q( g( W: |if (Cont & U_KEY)y--;) p+ z' l# Y3 H& G2 q- j/ i
    7 f8 N' Y) ?8 ?2 f
    if (Cont & D_KEY)y++;5 ]6 g6 |  l, F' P
    6 v  l0 K) @0 U0 d! c7 I6 U
    GsBgOffset(0,x,y);% v. W  H8 I& _* a; q0 }; z

    / ~3 H$ l8 t7 h0 b& x) |+ ~7 Hif (Trg & L_BUTTON)mos++;//当按下L键的时候,马赛克变量自增。这里没有用连续按键。因为连续按键的话,马赛克变化太快……
    2 Z# ^+ Z3 q- E7 L; ^* K2 m
    3 d8 v% V  P" d2 b4 V* j% N) L4 g4 ?if (Trg & R_BUTTON)mos--; : |4 y  [5 d5 Z' l
    6 @$ t* G6 Y* A+ B
    GsSetMosaic(mos,mos,0,0);//同样,这里要对键盘操纵的马赛克变量做处理。我们这次的DEMO没有用到精灵,所以后面两个参数均为0。
    % Z9 N6 l# L7 h& y
    + H4 l+ {8 n9 \  U}) b* f; J; O% K( r' G6 p' z- G0 N
    / q, M. {' e, _2 S
    这是效果。
    . N& ~/ b! t2 K" E( A
    * ~9 O; _# ?4 n4 `$ h
    : @# _9 g$ J; ?# r; K( M& H
    % j! F6 A6 @/ e' o' _1 `
    ; f) i/ \0 H/ C) B; p4 ]9 e: A& S$ Y4 j9 X
    注意,马赛克特效属于破坏性特效操作,就是说,如果进行马赛克特效之前不对图片的数据进行备份的话,就不能恢复了!这一点,相信大多数朋友深有感触……例如一张H图片,重要部位被马赛克处理了,那么想恢复原状就是不可能的。(众人:(鄙视的目光)只有你才深有感触吧……)幸运的是在GBA里面是有备份的,而且这个备份操作是自动的,不用我们操心……" ^6 Y7 L3 j3 b" R+ G6 k  l
    ( r6 A" B1 d0 J6 A# V: a
    有关马赛克特效暂时说这么多。其实灵活运用马赛克特效可以作出一些很出色的软件特效,例如水波特效。有关这方面的内容需要涉及物理方面的牛顿定律以及数学方面的正弦函数和随机函数,应该说比较深奥。到了《GBA游戏编程高级技术篇》里我会简单介绍这些软件特效的算法。- n2 ?# }6 {" s+ l0 A. D3 e! j. `
    - G; v; {. B& E3 t$ V2 Q- j
    接着是FADE特效。这在GBA游戏里面也是很常见。例如游戏开头的LOGO。我本人就很喜欢在游戏刚开始的时候显示我的LOGO,然后用FADE特效将LOGO变黑,然后慢慢变亮,停顿几秒钟之后再慢慢变黑……8 }+ P7 l+ e. \+ o* |
    * N3 p9 z4 \$ Z5 [
    来一个DEMO吧!还是用上面那个例子。
    . K5 J. Q- \7 ~; m2 X1 q3 C& m+ H# {1 X' Q5 b8 @* Z# o7 Q+ I
    #include <GsGBA.h> ( Z8 V; o" u- F' [' a- K

    8 T# X3 o2 a' U6 |+ X#include "BG.map.c" - [7 F% [9 ^& B. F
    / C6 a! Q8 m/ @, ^
    #include "BG.raw.c"
    1 M7 y/ d% g' n
    + e- {8 l. S$ I#include "BG.pal.c"
    6 c6 R+ q8 g. i$ O& {* i
    . y8 Q4 ]% f2 g) H, Qint x;int y;//这里定义了两个变量。这两个变量是后面对BG进行卷轴操作的偏移量。
    + e% o% j! j6 Y# I9 H& [$ f$ |
    " \# [! W1 _% }) V. O7 [int brightness=0;//这个变量是用于FADE特效渐变的变量。" L2 R6 k5 j; {) _9 [$ g' C

    1 y" {6 i& v* X% Rvoid Vblank();
    7 O7 B7 E1 I7 G/ s, u  x$ E
    ; w3 Q- \( K8 q0 [  s2 M; R/ }4 pconst IntrFuncp IntrTable[14]={7 \5 {! l7 U( V# R% L* B* M  R

    ' D. K8 A" ~- ^; W+ i3 c4 gVblank,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy
    2 G" n( J) b9 v  p
    ) }: I: D1 r2 n};
    0 n& s& a  Z! M$ C$ d, ?  z; \. q6 D
    int AgbMain()
    2 q# F- o0 `  C1 W8 ^& f" l{
      m& O0 O; z+ q' {0 z$ J, M, ~
    7 f, y* k4 x! U+ X3 N  \% I! VGsOpenIRQ();; h7 W  p/ q& m. p1 T  s
    , P: Q4 R# C$ D- m5 {* L, u  |
    GsSetMode(MODE_0);
    , d% \6 }, v, b, r4 _: E) }" S
    GsInitBG();
    9 j/ o. |4 M# `( ?" f
    " w$ ~! |1 k, _: e9 N' ]( [; \3 C6 k' NGsLoadBgPal16(BG_Palette,0);
    ' V- Y1 U1 C# n- o+ y7 a/ ?* g& V6 p7 e9 |. q9 o9 y; s
    GsLoadBgMap(BG_Map,sizeof(BG_Map),31);1 a" y' @+ V* V! I9 @$ v/ k" c# |

    5 ]2 P7 p4 m( }  s) WGsLoadBgTile(BG_Tiles,sizeof(BG_Tiles),0);7 `% O* C7 `  F! b  R8 C
    8 o& y+ p% N2 i% _3 p; y: ?
    GsSetBg(0,Gs_BG_TEXT_SIZE_256x256,Gs_BG_COLOR16,0,31,0);//最后一个参数是控制马赛克特效的,这次不需要马赛克了,关闭它。
    ) S7 k* S, A6 S% o! Q3 y4 A
    : b& u! y$ ]9 m  ~# I* ~; v. SGsSetEffect(BLD_BG0_1ST,BLD_BG0_2ND);//同样,设置完BG之后一定要记住打开特效支持!这次还是只针对BG0做特效,因此这个函数的两个参数均选择BG0。% l; o' o+ L" K; i

    * @4 Z3 N$ l9 A/ ~GsBgPriority(0,0);
      K( i# \, L; O+ b) f+ x
    $ r, o. o; i9 |7 [$ NGsSetBgState(0,ON);
    ) q1 @9 ?0 d7 k% c
    5 _" W! P( }' j* l. u( ]# G5 cwhile(1) # o0 c9 v' o' w
    ) m1 a6 \9 R# ?! C" R
    {
    $ `, I# L, X, l  n, @4 X  I4 C+ ?+ ~2 j9 T4 l9 z% {2 s8 f
    GsWaitSync(); 6 D  g  I5 i% ~% A& x- w$ |

    * w6 M7 g: A  x3 b# O, X( I}
    4 I7 c9 S0 q3 N$ l/ u0 F- L/ A
    & g* W! y# W% X6 i- ]* E6 Ireturn 0;
    . H# c9 q7 I7 u9 Q: [6 X' o, D9 q) |  h
    }7 k8 r) D9 t& @% F8 b$ z8 H$ q
      ^( T$ ^+ [% t  K8 Q/ z% G
    void Vblank() : Q! R3 m! ]* G2 r1 t

    ! V- t# c" {2 w{
      V, \7 \0 k0 r* H3 \+ W* i1 V  T6 p
    KeyRead();
    $ W+ R9 |0 g2 z2 _+ ^6 L$ M* n3 h& X" C
    if (Cont & L_KEY)x--;
    8 b2 J( s# K3 {( o/ e; W5 g! @( ^
    8 J, s( R. |+ D- jif (Cont & R_KEY)x++;
    : T/ ^! i+ v8 }; D5 s0 s" B, T1 r* S0 Y0 a
    if (Cont & U_KEY)y--;; n& `7 ~3 K7 {0 N* a: G
    . l4 ^- [- M. o
    if (Cont & D_KEY)y++;
    # \6 }$ `7 `: s7 a9 |0 B8 j7 b$ x: O+ J' _8 @& ^
    GsBgOffset(0,x,y);' Q: N: t+ e9 a/ Q0 R2 u( W- T) @
    5 Z. f& O& P! e  w. B1 M
    if (Trg & L_BUTTON)brightness++;//当按下L键的时候,亮度变量自增。这里没有用连续按键。也是因为连续按键的话,亮度变化太快……
    ; e' r$ M8 L; `: ~$ }$ p
    ( ~( H% F  s4 j' V% O. Vif (Trg & R_BUTTON)brightness--; * G- Q$ P& @$ m6 g+ ^

    1 ^2 _% T# ~' m$ T5 g# oGsSetBright(brightness);//这里就直接控制亮度了。要注意的是,帮助文件中规定的亮度参数是s16类型,也就是允许负数。但是即使你把这个变量定义为无符号变量程序照样正确。原因很简单我就不说了。另外帮助文件中规定参数范围为-15~+15,其实超过这个范围也无所谓……你可以试一试!$ o! _9 N: U: r
    " u7 v" [5 V" u( T+ Y; X7 ~' q
    }. u& N, g% ^& {" G# h( n

    0 H1 @; r4 u. g8 d3 x效果图:
    $ I0 g1 R  J) C
    * Q& d' c* f8 s/ ]4 W6 ?  ~; x' K4 K1 R  V

    % c/ m; e6 d$ w2 U
    8 s8 c4 G% `: T0 X1 N3 ^
    5 H/ @" ?* S2 P, y% OFADE特效也是破坏性特效操作,不过你仍然不用操心……嘿嘿……
    # v; Q5 E3 f& Z" k& `. Q0 `% [# H; z5 J7 ]8 Y1 j" Q5 D
    利用FADE特效,可以很方便的做出黑天特效。GBA超级大作、人世间最强大的GBA游戏、不到长城非好汉不玩这游戏就白活的经典、人类智慧最美丽的结晶、(后面省略形容词10000字)的游戏《黄金太阳2-失落的时代》中,有一个这样的剧情:主角们旅行到一个狼人村庄,刚走近村庄的时候,天暗了下来,然后迅速天黑了。(很令我奇怪,无论我走了多长时间总是到了村子口天就黑,即使刚从客栈里面爬起来……这几位兄弟时间掐的还真准!)这里的特效就是FADE特效了……当然,也可以做出刺眼的光照特效。仅次于GBA超级大作、人世间最强大的GBA游戏、不到长城非好汉不玩这游戏就白活的经典、人类智慧最美丽的结晶、(后面省略形容词10000字)的游戏《黄金太阳》的游戏《口袋妖怪》中,神兽古拉顿觉醒之后,整个芳缘地区开始出现烈日天气。那种刺眼的阳光,就是FADE特效……
    8 Y$ o2 c. ^6 J' E9 e: s5 r) s: W
    5 D/ ^2 S/ g' n$ o. f+ N" {* O7 P: m然后是窗口特效。顾名思义,窗口特效就是BG上面出现一个区域,限制了BG的显示范围。我孤陋寡闻,玩过的GBA游戏很少,迄今为止只发现了一个游戏使用了BG窗口特效——《RPG制造》。游戏制作中有个选项,就是游戏结束后出现STAFF表,是以什么方式出现。有一个选项是背景图片被挤到屏幕左方,右方空出来的区域,显示STAFF。这个显示过程就是窗口特效。8 {" K3 l/ Z& n6 b, Q& g
    1 R. S* }  w- s
    GBA硬件共有三个窗口可用:窗口0、窗口1和OBJ窗口。其中窗口0和窗口1效果相同,呈现为屏幕上的一块矩形区域。OBJ窗口可以算是OBJ的一种特效,它只能配合OBJ使用。一旦OBJ使用了OBJ窗口,其显示就会变成一个轮廓。关于精灵窗口特效,最经典的例子就是GBA超级大作、人世间最强大的GBA游戏、不到长城非好汉不玩这游戏就白活的经典、人类智慧最美丽的结晶、(后面省略形容词10000字)的游戏《黄金太阳》中,角色们在大地图上使用“Psynergy”(好像是被翻译为“精神力”的吧?),其身体就变成了一个轮廓……还有,在阴影里面使用Psynergy“阴影”,身体也是变成轮廓,而且很明显……
    ( H+ K6 k& |4 x: _8 `
    $ y7 O& \( ^' N3 z% d( A+ n. \6 K我们也来一个窗口特效的DEMO。2 O3 Z* u+ f, S9 ~% W" A" Y
    3 V: O$ r7 F+ l$ q! E
    #include <GsGBA.h> $ J0 T3 o7 @0 ~  W) B3 |2 S0 Y
    0 ?8 k7 }8 T: n6 F) k- _1 v
    #include "BG.map.c" % }1 y% a" ~" V; Q

    7 V8 X8 w& O7 n/ G#include "BG.raw.c" 6 Q3 y: N# e% {+ p2 Z

    ' b6 L, {6 S, c4 r/ d#include "BG.pal.c" + q' C+ w" m+ @: v

    1 f: `. F0 k/ j3 K; T, q4 s#include "Blue.map.c"
    ! h% k/ V8 f# _
    2 x1 a2 f( o" V3 r+ \  x, j/ V5 Oint winx=0;. \9 ?0 M7 J' j9 E8 Q$ X
    + w" x1 V8 S# f/ d2 G5 `' K
    int winy=0;//这2个变量是用于控制窗口大小的变量。; q5 B  }, U# n

    9 `7 a% z6 n. Avoid Vblank();
    3 z* `; \$ }! X9 [" A
    # ?: `$ H1 O; x" Rconst IntrFuncp IntrTable[14]={
    0 T6 s2 J" Q3 |
    6 ~. s0 Z1 Z6 O  x% z+ aVblank,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy
    ( X# F6 }! E) |1 c6 B: i# D' l
    , }9 B9 T; C& A" q3 a0 x7 X};
    " ^% Q3 u0 o. ^" E! H, G( a4 q& Z" _' V5 O2 p9 d4 a9 \3 Y8 @1 Q
    int AgbMain() 6 U% W  j$ d: G" Z
    {
    , B0 K$ o2 P: X& ?# D' F: \3 O3 ]
    " S. Q; u# S9 {4 FGsOpenIRQ();$ u  k8 J0 ~" r; K' n+ U) k9 R

    7 N% _# q, w: I9 o: Z/ d) ]GsSetMode(MODE_0);
    1 J, C8 M) q3 C6 H# s: S2 Z, G1 w! D
    . t. j$ h8 g. y2 U) M0 [GsInitBG(); ) Q& u9 a2 d0 p% w' @* y

    ( t0 Y" y& V+ a4 iGsLoadBgPal16(BG_Palette,0);1 b$ K* b4 U, {- h% n

    * G% p) Z0 Q0 n2 `+ AGsLoadBgMap(BG_Map,sizeof(BG_Map),31);
    $ h7 o3 y1 B( z$ s
    2 B; m! M$ C- P7 V( }GsLoadBgTile(BG_Tiles,sizeof(BG_Tiles),0);" x; w) E3 J+ h+ ?& |0 U
    1 R. ^7 v; V9 n8 ]
    GsSetBg(0,Gs_BG_TEXT_SIZE_256x256,Gs_BG_COLOR16,0,31,0);! M7 u- P" C6 O7 T4 B

    8 P8 W: i' K+ N% d7 c+ J* m; M4 @. Z) `( ^GsBgPriority(0,0);
    ; t  K- @( a$ r  a" l5 O) o8 H! X2 F. \& S% ~# f" v9 A) }
    GsSetBgState(0,ON);
    ! k) p- m* }# |
    - U7 {& }0 D- Q" Y: LGsSetEffect(BLD_BG0_1ST,BLD_BG0_2ND);//同样。
    ) x7 b$ u- T/ \( y9 p  X
    / J9 y) F  q$ L+ I) x; AGsSetWinState(0,1);1 K! O* Z, i& B6 v. o

    ; [/ z, X: V7 U; WGsSetWinState(1,0);//设置窗口状态。我们没有用到窗口1,只用到了窗口0,所以激活窗口0,关闭窗口1就可以了。* s( A- ~/ P3 G, O4 p' J* S) h$ E; Z# q
    ( y& v. O: T1 {
    while(1) ' [% a& i9 \( E( t
    7 S4 S7 U+ Y  L' ]
    {
    9 G" _# [8 |$ K0 m  I5 A4 \
    & M! c% A6 q$ k* ^# V6 KGsWaitSync(); 7 j# L3 }7 m& X# e5 a! f( [8 h
    / `2 B" a& `4 [0 j
    }
    8 [( H$ ~2 g0 _! a
    3 ~" G: j* P/ o4 ?  Z: B" Freturn 0; ; ^) p  {% ?! D# N7 V) v# b

    5 e! h, O  K! h$ D* R2 e}
    + ^) m" j% ^. m$ T$ \5 S6 r  S1 t9 n1 T1 k+ @
    void Vblank()
    - L3 _) ?2 }8 K) u- k- o/ T- Q6 J: ~+ a
    {7 ]' |0 Q: n- @8 C1 p& r
    - a% S8 {* C( g  |# z' s1 y8 }
    KeyRead();8 ]* O  f3 z2 f

    9 }0 L# _' t! nif (Cont & L_BUTTON){winx++;winy++;}//这会有什么效果,你能猜出来吗?+ b8 j$ s) Q( k# p$ V, {1 i
    " F8 B& W% J4 M( N6 i: n; T
    if (Cont & R_BUTTON){winx--;winy--;}
    0 M3 I+ i  A7 Q
    9 l) y, @. g1 w2 r- TGsCreateWin(0,120-winx,80-winy,120+winx,80+winy,WIN_BG1_ON,WIN_ALL_ON,ON,ON);//创建窗口。使用窗口0,大小由我*纵,窗口里面显示BG1,窗口外面全部显示,允许窗口特效。BG1我们没有用到,所以就是“空白”。
    . n" x' U4 H/ w  B* w( a
    # W2 q6 ?5 h, ~9 G% T, r% \}
    0 t" F5 M' t& M, T2 R/ O5 E+ Q+ S7 u3 @, ^
    这是效果。7 E' i- X2 H' H9 G

    ( i9 Q, {; q& f* L8 d, {  V
    + ?6 \- W5 k' u* K+ f: ^, y" Q% E2 h- ^% l5 N

    ! W" A. E( [' V5 n( c4 [+ d3 a) X) Q& r) J4 I8 D/ R2 y9 ^( s- }2 O) ^
    窗口特效不是破坏性操作,从宏观上看,其实窗口就是一个“虚拟”的BG,这个BG盖住了真实BG的一部分……形成了像上面的那种效果。
      }: n7 A9 ^  M, v3 ?9 ^) j6 f6 V  r
    在GBA中,存在两个窗口,就是窗口0和窗口1。理论上讲这两个窗口都可以由你操纵。不过,或许是我的分析有误吧,我只能操纵窗口0,窗口1操纵的时候,总是出现异常的状况,所以,有关窗口1的资料,我暂时无法提供……召唤达人中……如果你要使用窗口1,请慎重!
    & J( w1 K% ?8 `' R% ]3 Z, |
    0 y, |) r1 f9 r( [5 n最后,剩下Alpha特效了。大家都知道“三原色”——红绿蓝。这三种颜色通过不同的比例混合,可以做出各种各样的颜色来。GBA硬件可以完成这种颜色混合。这种特效,就叫做Alpha特效。7 h! r; e8 i4 F' v

    3 C4 S) h9 w0 L6 M+ i这种特效在GBA游戏里面不怎么常见,我只见过一次——还是在GBA超级大作、人世间最强大的GBA游戏、不到长城非好汉不玩这游戏就白活的经典、人类智慧最美丽的结晶、(后面省略形容词10000字)的游戏《黄金太阳》中,片头黄金太阳升起的时候,背景的阿鲁法山在阳光的照射下显得颇为壮观……阳光穿过浓厚的云层射下来,形成美丽的光环。这个光环,就是活用Alpha特效做出来的软件特效——光晕特效。我个人认为这是所有游戏里面最令我陶醉的一幕画面!这个特效,把这一景观做绝了!+ C) N* M" S# N, e  z- \
    1 j; e: A$ D' ~! @: U! E2 T
    1 E+ v1 [1 \; M* p

    5 u# Y1 ?8 [6 P- y6 w6 G: S5 V
    4 M+ [2 d) t$ ]# ~
    5 l8 c$ H( z) a0 W4 t* Q3 m关于这个硬件特效,我不想写例子了,我觉得,大家参考samples文件夹里面的DEMO,自己试着写一个就可以了。还是那一句话,用的好的话,这个特效是很炫的!7 w" d+ c+ S7 W% u6 k- b2 ^! n

    - Z6 n) [+ X: ]! M$ P7 {到现在为止,有关BG的内容就全部讲完了。不过,这并不是说BG就这点东西,更深更好用的函数还有呢!下一次,将是《GBA游戏编程菜鸟入门篇》的最后一次教程了。学完这最后一次教程,你就不是菜鸟了!因此在中级教程里面的例子,我不加注释的部分你应该能够看懂。如果你还是看不懂的话我劝你还应该再从头看一遍。下一次的教程,GBA上面的音乐播放!学会这一部分,配合特效和多层BG,做出来的效果不用我多说什么吧……嗯……还有一点小小的事情:如果你有点五音不全或者对音乐的鉴赏能力很差,那么你就不要学下一次的教程了。因为下一次的教程要涉及自己写MOD,这是和写MID非常类似的工作,如果你连MID都不会写,那么MOD你也未必能写出来……2 I) [$ z7 O& f5 H0 g0 C) U6 F, N+ Q7 X

    , y, m; s0 L. o2 o9 w6 yGBA游戏编程菜鸟入门篇(七)——GBA音乐播放
    % L" S' K& T1 s$ j* B# z. K( r! H
    . [5 U! r& f( V8 `
    4 b* s/ t: F$ t( K7 h' b5 @GBA上面一共有6个声道,分别为SOUND1~4,和DirectSound A/B。这6个声道硬件上到底是怎样播放声音的,这个问题实在是复杂,官方开发包里面的帮助文件和寄存器映射表我基本上是完全看不明白。不过,GsLIB中提供了一个非常好的工具——MOD播放器!这个工具,使得我们可以非常容易的在GBA上面播放声音,无论是背景音乐还是效果音或者真人语音……不过,会有一点点的失真。
    ; ]6 _3 |$ W7 l3 Z% T) j
      l4 h( w6 [8 a3 W9 l" z8 |: T在GBA上面播放音乐有两种途径,第一种是使用GsLIB2.6 ARMSDT版本提供的工具——ModPlayer——来播放MOD;第二种就是只有GNUPRO版本的GsLIB才能实现的,使用MP2K播放MID。ModPlayer有种种缺点,但是由于使用简单,我们还暂时不能将它抛弃;而MP2K才是终极之道,官方游戏,大多数都是使用MP2K的……如果你掌握了ARMSDT版本的GsLIB的使用,那么我强烈建议你转向GNUPRO版本!因为两种版本中的函数基本上都是一模一样,相信你会很快习惯!6 j8 S" s" t& U* s

    " b' ^$ T6 F8 H$ V; }, z5 T- @首先简单解释一下所谓MOD。MOD是一种音乐格式,和MID类似,只不过自带采样数据。帮助文件里面就是这样解释的。详细的资料,我也不知道,你要是有兴趣,可以到网上找一下相关的资料。在我们这里,了解这么多就够用了。3 r4 x" U! Q7 H# H
    5 X$ O, y. f' @6 w; h/ ]8 u6 r
    既然MOD自带采样数据,那么MOD实际上就可以用于播放WAV,只要你的采样数据是你所需要的WAV……因此,如果你想在游戏中加入真人语音或者效果音,只需要录制好你需要的声音成WAV格式,然后在ModPlug里面导入你做好的WAV成采样数据,然后……
      \9 Y- v9 N6 {' y+ O" f' }
    6 ?8 n; I) U/ q2 q首先说明一下,制作MOD的软件应该很多,但是我只找到这一个;不过这个软件也的确强,虽然做出来的MOD有点失真,但是这很可能是由于我对软件生疏,制作的时候造成的……不能不说这软件真的很好……由于我也是刚刚使用这个软件,操作的时候出现的错误在所难免,请各位见谅,雷精灵也就只能给各位提供这些资料了……当然,如果您很擅长使用这个软件,那么还请您多多赐教!
    ' R( {7 D  p: ?* ]3 B) g1 L7 m9 H& }0 W- h: a; E
    哼哼……我们来实际操作一下!3 V6 [4 {3 j! u! D+ k( w: ^

    5 x# s8 Z! l7 b) |* b$ r首先是ModPlug的界面……; L% w, @0 G$ n% ~6 V7 ^$ L
    6 F4 n' \1 G% a3 x& m4 R
    4 k! |& E# F, Y- G
    这个时候你可以用它打开一个MOD文件,熟悉一下每个部分的功能和作用……关于MOD文件,samples文件夹里面有两首很好听的MOD……  k9 @2 m! |, }% D: U% r9 J* }* D

    / [, U! n% h8 P6 w1 y+ n  k0 B( R. |自己录制一点语音吧!我从游戏里面截取了一点声音并转成了WAV格式。这是PS《生化危机3》钟楼里面的八音盒音乐,很爽……不过要注意一下,转成的WAV,最好不要是立体声,我不敢保证立体声的采样数据能不能做出MOD来……) g2 ]5 }' l" u: V  Y

    ) C: |- i/ q; V' l6 |# i点击这里下载《生化危机3》钟楼八音盒WAV音乐:http://www.tgb.name/index.php?act=Attach&type=post&id=516
    / |* }* |: a) d$ c% Q/ |1 H
    . p9 {9 z2 ?) o& q, r然后打开ModPlug,新建一个文件。7 N7 `" n. s7 L4 p
    9 e* T1 \4 ~* Z. u8 P
    然后主窗口里面出现了5个选项卡,分别是General、Patterns、Samples、Instruments和Comment。大家的英语一定都比我强,我就不翻译了……
    1 O) Z! d7 R* K9 X+ C' U3 w0 A5 N3 V& P1 e! D* y3 d1 P6 j
    点击General选项卡使之前置,最右边是声音的幅值显示器。这个东西大家一定见过!那两路显示,代表声音的两个声道。
    # B' F5 m' W7 ~$ F3 m
    7 _& s8 e& d/ `6 \- [: {8 ~. Q关键是它的旁边那个“Change”按钮。点击,弹出来一个窗口。把Channel数改成4。这样会减小MOD的体积,而且由于是在GBA上使用,4个Channel也完全足够了。
    " A) J! z, O: A  Z
    / w; M3 ~/ Z. s然后点击Samples选项卡,这个时候采样数据前置。5 p; V3 H$ j7 {
    / X/ X9 j2 d6 y) C2 f9 ^6 ?& p

    " N( c9 a/ r5 ]4 L- m7 l1 a  k% s
    " K: X( S# f; X) h
    直接导入采样数据,也就是你准备好的WAV文件。导入完成之后下面的波形窗口就会出现采样数据的波形。点击播放按钮会播放采样数据。5 E+ x; w& t: R! x. E: ^; y/ e

    % R* ]$ E: n# L: k4 f下一个步骤非常关键!请务必照着教程做!$ n" B3 [0 g0 D* U' Y' k) I; k

    + k. j% l8 Y4 p8 C- g' O# {0 e, c/ X% n) H! {: X8 T+ [: f

    % w, _  \! w- i; d9 Y4 A- c
    7 j  R/ Q7 j: p( L$ b& v) i2 n* g3 g( H+ o* ]1 R. B7 A
    点击两次那个“Downsample”按钮。这个时候下面的波形会有些许的变化。8 q, Z3 B: J) b4 C

    " e( z/ u% J6 d6 k. x$ X然后点击Patterns选项卡使之前置。准备写音符。+ U: |$ N" Q- d  S+ B

    . {( x: J, o# A$ @+ H# }双击第一个Channel的第一个音符,弹出音符属性对话框。音符选择“C5”,采样数据选择第一个,就是你刚刚导入的采样数据。由于我导入的采样数据比较长,我新建了一个Pattern,这样可以有效的防止声音被掐掉尾巴。8 h: p6 d5 a- z5 x
    / M3 P. ]+ Z& k, E2 M
    然后直接另存为MOD格式。7 |& M0 n4 [9 P' b" B

    $ d' `) ?6 D% I$ z& o: Z6 G, a关闭,再打开,听一下效果吧!虽然音调有点低沉,速度变慢,有点失真,不过效果还是颇不错的……如果你想做成立体声,就把另外几个Channel也填上音符C5。4 m0 z- t* P" F: O( k6 X
    9 L# F8 {0 K, w) ~9 ~7 n
    如果你对这个失真很在意,那么我建议你先用变声软件处理一下你的采样数据,然后再导入……
    & I/ X8 ]! b& b
    9 \' r, d3 t/ S; T+ O( a: w5 f那么,MOD有了,开始写程序吧!. l* t. c$ b; j! c9 i# o
    & x$ {& E7 ~: T# r. q1 @2 T
    (众人:嘿嘿……这还用得着你写?光看帮助文件和samples里面的例子就能够写出来了……)% u0 o2 |  V2 s5 o9 c$ P, y
    ) h  l+ B8 ]( Y3 T, y$ t. ~
    雷精灵:(尴尬)啊……那……好吧……我就把我写的DEMO放上去得了!
    " D& S0 X" k( A. O! i8 u* m9 B$ u4 ]
    点击这里下载DEMO:http://www.tgb.name/index.php?act=Attach&type=post&id=518# ?. _  X& Z( V: k0 C5 G1 Z% t
    7 f" X* A5 [* K0 p+ }
    就像这样,GBA可以播放音乐了。只要更换你的采样数据,就可以做出各种声音的MOD。如果你导入两个或者更多采样,写音符的时候在其他Channel使用别的采样,那么,效果就不用我说吧!如果在其他Channel使用相同的采样数据,但是有点延迟,那么就可以做出回声效果……只要你能想得到,就不怕你做不到!2 @* G% F4 y. c

    ) s% }* B) J! h0 y- U同样,如果直接使用现成的MOD音乐,就可以当作是背景音乐;录制效果音作为采样,那么做出来的就可以当作是效果音;录制语音,那么就可以实现播放真人语音……有没有人放一首歌上去?诸位不要笑,完全可能!% _6 B5 h! S* w# F+ P+ u6 W

    ( c+ S" i8 _2 Y, @5 u7 d到这里,初级教程就完全结束了。现在的你,已经不是菜鸟了!毕竟,能够自由的操纵BG,随心所欲的制造特效,还顺带着加上背景音乐,已经很了不起了!下一次教程,我们就涉足更深的内容——精灵使用!精灵技术是游戏中的一项革命技术,掌握了精灵使用,你就掌握了GBA游戏编程的另外三分之一!
    ) ]  e+ y3 a& z, ~  A4 y. t
    ( R0 u3 ]5 C2 r7 C2 f: GGBA游戏编程菜鸟入门篇(特别篇)——制作一个MINI游戏
    9 H, q8 h/ y  x4 i! W

    $ _* `+ T. r/ @! U  r初级教程中,我们掌握了TILE模式下BG的简单内容,学会了使用多层BG,能够使用简单的硬件特效,也可以进行音乐播放。这次的特别篇里面,我们将尽量多的使用我们学过的所有知识,制作一个MINI的游戏!8 ^' H$ o& B5 N1 d
    0 [! G* g% V6 R! H, r8 @* W
    一般的游戏中,哪怕是很小的游戏,“精灵”技术也是必不可少的。不过,我们还没有学到精灵技术,还暂时不能使用精灵。但是,极其简单的游戏中,如果活动物体很少,并且不需要进行大幅度的变动,那么也可以用BG代替精灵。这次我们要做的游戏中,就用BG代替精灵。
    , T. L! e3 @9 O3 \# }5 i$ I' R. a: v0 O4 B9 H3 [
    做个什么游戏呢?HGAME怎么样?* Z8 Q* A! k* _. j. ~: n

    3 A' {5 i# }6 y) ^: G9 L0 D(众人:(怒)你个色狼!). h" u4 j& d" J, @: h0 T, u

    ) Q& ~- x1 V8 {2 E8 F雷精灵:大家不要误会呀!HGAME一般注重情节和CG,活动物体和精灵一般不多。所以,这个时候选择HGAME,应该是合适的……% Q- L* C) h+ U2 u8 A

    , q. @$ z2 D% c5 u6 X我最近在网上看到了一个挺有趣的HFLASH,大概的情节就是,游戏随机出现一个数字,然后下面的秒表计时。如果你在恰当的时间按下按钮,使当前秒表显示的数字和给定的数字相同,那么游戏里面的女孩就脱一件衣服……直到你把女孩脱光……相反,如果你没有准确的在恰当的时机按下按钮,那么女孩就穿一件衣服……若干次失败后,就GAME OVER啦……2 F; r1 b2 {/ _$ X( V  o3 o  z9 v1 g

    5 H/ Q. C- \4 T8 O1 U/ ^1 H2 R如果要把这个游戏在GBA上实现,应该怎么做呢?0 s2 k  w1 R* g: P
    9 W( b+ A1 q2 m9 D% l3 y4 F
    1.随机出现的数字,可以用随机函数确定,数字,以文字的方式出现,可以用MODE0下面的文字函数实现。不过,我这里最好还是先说一下,关于这个数字,最好还是用精灵的形式实现。因为用精灵的话,你可以将文字做得很漂亮……不仅如此,精灵实在是太灵活了!用好了精灵,效果真的会相当出色的……3 o6 p  D+ L0 r- D; R+ }

    / c. _: |4 }6 J( F" h) k2.秒表,因为频率是固定的,所以可以用中断实现。VBLANK中断,它的频率就是固定的,非常接近60Hz。秒表显示的文字,当然也是用文字函数。
    & K# _3 O8 @8 H3 N# N7 C% m/ C- ~$ n+ @* B6 [
    3.按键响应,也要使用中断。检测到按键的时候,暂停秒表运行,判断两个数值,然后分支。6 i- P0 x1 m. Q5 v

    * j( E* n. f( q* }6 w0 l  n4.女孩的图片作为背景,根据分支确定背景的显示和替换。
    ( z5 f, F2 [* @7 a% t: H4 N9 e9 g* D- v6 I3 K
    嗯……大体就这么多了。开始吧!
    % g5 V0 r6 ~  s: O" d. m$ {% y+ u+ W' V# ]# A- r0 N
    首先准备图片。这个游戏的所有资源,在教程的最后我均提供压缩包下载。9 e% U3 B+ |6 D2 N8 s( G! ~  d; o
    ' ~$ w+ K* Q$ A. m+ d; \
    CG,BMP格式,256*256像素,256色。因为处于BG最低层,不设透明色。" S3 b8 [7 S) o
    ; y% i4 L! \- ]% A8 J5 l
    文字层1,用于显示随机出现的数字,还有一些对话什么的。调色盘使用16色。, r+ T4 J. H' o$ s
    5 o# L+ U+ r7 L9 j0 B
    文字层2,专门用于显示秒表数字。调色盘也是使用16色。3 O7 W1 N, x5 G2 C4 u/ H

    # ~" o" @1 `' p为了出现比较好的效果,加上游戏片头。也算是为自己宣传了吧!游戏片头使用256色图片,256*256像素,使用FADE特效。* o" a8 C1 O7 P% A

    " N' k$ b$ Z, @% r- l9 F游戏标题也加上吧!使用马赛克特效。1 R, _) C7 y  ?4 `; a' e

    9 Z- V& V; f( ?+ l. W: b背景音乐,毫无疑问得要啦!格式为MOD。
    1 ~6 H8 w. _/ T) t3 o
    + F8 ^" p' K; J5 N4 y程序开始。包含头文件,包含资源文件。6 p( j) B( d7 i0 P- v

    3 y- `! c4 Y! O4 j7 ]* R% [主程序,打开中断,然后直接进入死循环。所有的工作,都留给中断去做。* `1 v1 b# n8 o6 ^! v7 p; h
    * [& I9 i1 P; I3 L  }& _: O
    由于中断要处理各种不同的函数,所以需要在程序中变换中断服务程序。有两种方法可以实现:函数指针和分支。
    4 h: l5 O' I' d  I" u+ _8 `9 q+ H+ t4 z1 L$ N
    函数指针方面,我没有学过,所以我也没办法讲解了。我就用分支来处理这种情况吧。顺便一说,用分支并不是一个好方法,如果游戏比较复杂,那么分支将会非常庞大;到时候恐怕连你自己都不明白到底是怎么回事了……; @$ P" t9 w( x/ V

    ) m8 S8 Q% R  }所谓分支,就是通过一个游戏流程变量(我一般称之为“剧情变量”)来控制游戏当前状况。这个变量为全局变量,初值为0。中断服务程序就是通过这个变量的值,来选择不同的中断服务。就拿咱们这次的游戏为例,游戏刚开始,显然要显示你的LOGO,因此在中断服务中,case 0的时候就是载入LOGO的数据,然后等待同步。中断完成之后顺便让剧情变量增值一下,那么下一次中断来到的时候,执行的就是case 1的服务。很明显这个服务程序就该是FADE特效了……
    6 E! k( |6 V9 W
    - L$ G% N9 H6 U# _' @1 S' @代码中出现了一些有趣的技巧,如果你不明白,不妨把你不明白的地方屏蔽掉,重新编译看看出现了什么效果,或许你会知道我为什么会这样做。
    & A. ?/ a* `3 b; P
    : [! M# P+ D! s# y: F; I游戏中出现了一个明显的问题,就是背景音乐。背景音乐在标题菜单中播放效果非常好,但是一旦游戏开始,数字开始变化的时候,音乐明显出现“卡”的现象。这是怎么回事呢?这与我们写程序的方式有关系。我们的主程序中,什么都没有,所有的工作,都是交给了中断去做。中断,顾名思义就是中断主程序,转向服务程序。但是音乐正在播放,这个时候中断主程序而转向服务程序,音乐自然也被中断了。不过,由于服务程序非常短小,所以中断服务被迅速执行完毕然后退出来继续执行主程序,也就是继续播放音乐,所以,从宏观上看,就是音乐出现了一个短暂的中断,从而出现了“卡”的现象。( [0 H. T, G) ^# [  G/ L- j) ^9 o

    3 f3 N4 |9 o" Y" w# O那么怎么解决这个问题呢?没有别的办法,重写程序,让主程序不要吃闲饭,尽量少占用中断,非得占用中断的话,尽量缩短中断的长度。如果非得大量占用中断的话,那就只有抛弃MOD PLAYER,使用MP2K了!) }' e: ^, l$ n$ @1 }5 z" m6 ~* D

      A- y9 S' A/ {( C0 y程序没有优化,因此冗余代码很多。你要是有能力,优化一下会减小ROM体积。程序中还存在一些预留的入口,利用好了,可以做出“秘籍”、“隐藏要素”!当然,你也可以继续扩大程序,加上其它的功能。比如纪录通关的按键次数,然后用存档函数保存数据;新写一个界面,做成排行榜,然后利用排序函数做出排名……毛主席说过:人有多大胆,地有多大产!虽然这话有点唯心主义,但是可以确定的是,只要你有想法,只要你有能力,没有你实现不了的东西……
    , _6 r2 {  l. Y0 a$ y# R
    4 O# S. |( R7 D

    4 F- x( E5 g+ v$ y: Z: X% |. o- x7 J, ~% a, R5 |2 O4 z$ v
  • TA的每日心情
    激动
    2013-12-13 22:59
  • 签到天数: 346 天

    [LV.8]以国为家I

    发表于 2011-9-13 11:50:15 | 显示全部楼层
    太长了!顶HOHO!
  • TA的每日心情
    开心
    2021-1-29 11:50
  • 签到天数: 104 天

    [LV.6]王国居民II

    发表于 2011-9-19 18:45:14 | 显示全部楼层
    就是太长了
  • TA的每日心情
    淡定
    2014-1-31 21:15
  • 签到天数: 435 天

    [LV.9]以国为家II

    发表于 2011-9-20 23:04:53 | 显示全部楼层
    {:soso__5124458142922744482_4:}难道汇编就是这个的应用版?
  • TA的每日心情
    淡定
    2014-1-20 22:59
  • 签到天数: 156 天

    [LV.7]王国居民III

    发表于 2012-1-27 14:04:55 | 显示全部楼层
    叉烧包······
    9 V2 K2 n4 {) _7 S8 Y没图真晕啊······
    您需要登录后才可以回帖 登录 | 加入王国!

    本版积分规则

    关于我们|联系我们|口袋王国|Poke The Kingdom.

    GMT+8, 2025-2-19 06:59 , Processed in 3.132340 second(s), 18 queries .

    Powered by PokeTK V5

    © 2011-2021 Poke The Kingdom.

    返回顶部