TA的每日心情 | 别闹 2013-12-11 16:57 |
---|
签到天数: 129 天 [LV.7]王国居民III
|

楼主 |
发表于 2011-9-13 09:23:29
|
显示全部楼层
GBA游戏编程菜鸟入门篇(五)——Char Base冲突
, R7 b, G3 }! b9 V( O% k) R% g- \( k. Q5 Z
上次的教程实在太长,我的手,打字都快打抽筋了……帖到论坛上的时候甚至发生了“发帖内容不得超过15K”这样的警告!我不得不把教程分成两次帖上去。但是这同时引起了代码错误,所有颜色都丢失了。这是没办法的事情,唯一解决的途径,就是让管理员将论坛限制放宽一些……扯远了……这次的教程很简单,内容也少,不过,不是很容易理解……大家做好心理准备!
; H p: N* |. ^& |1 G/ v
0 L9 @+ c1 r$ T! G z. n( ?想必聪明眼尖的朋友一定已经发现了,上次生成的那个DEMO中,256色背景图片,楼房窗口处和源图片不同。察看TILE VIEWER,发现原本用来存放Frame对话框图片的CHAR BASE,出现了其他数据!0 D& p& L* p0 h4 g0 @ S( \" |7 u. G
1 _2 V L- E7 Z. ?7 b; z* R4 V# Y% X& m
7 ^7 f8 F: R6 T7 B: D; {
_3 F8 T: b9 {& _
3 T7 a. k6 H }& @8 g: j5 }
G5 Y2 q0 p+ a5 B3 E9 |' V
用16色察看,只能依稀辨别出来,最前面几个TILE还有点像Frame的某些TILE,后面一堆TILE,就不知道是什么了……用256色察看,竟然发现这是256色背景图片的下半部分!
& E( Z a3 B- _ G2 `
2 L% O! |& T4 i这是怎么回事?
1 f/ q: @ w, R/ g, O
2 L+ c x. k. a+ a& m3 N回过头看看程序:背景图片是存放在0号CHAR BASE,Frame存放在1号Char Base。存放在0号的256色的背景,怎么会跑到1号里去?
. L$ u) P. `$ g3 {- }/ f# O1 D) A9 p$ x9 U
(众人:0号Char Base不够存放背景图片,导致溢出的部分跑到了1号里面去啦?)& Q/ ?* @" R0 J! ?3 n9 G
$ ^: v1 R3 t5 k9 j
雷精灵:强啊!你怎么知道的?
$ H `% K; A8 O- n7 ?% C
, C9 ]" ]; q0 |, D(众人:把两个图片的TILE文件打开,明显BG的TILE远大于Frame嘛!而每块Char Base的空间只有0x6004000-0x6000000=0x4000这么大,万一BG的TILE大于Char Base的容量,那很明显多出来的数据会顺序存到下面的Char Base里面去啊!)4 E8 n! F! k% n5 f$ ~6 j
7 \9 Q* @- \" Y6 d, W5 B雷精灵:Exactly!那么,怎么解决这个问题?/ _9 T7 ? B: e% s
# p* a, |$ m% v7 o% B/ d
(众人:……)
4 \6 A$ [/ L+ C
6 h7 [9 ?% G4 K0 T$ U雷精灵:(得意地笑)最简单的方法,就是不用1号Char Base存放Frame图片了。根据程序,0号已经用来存放BG图片,1号被占用了一部分不能使用,2号存放文字,那只能用3号!: C+ I$ m# J4 G
4 N# X" m' X4 ?- e; } G# e(众人:慢着!3号里面有这3个图层的MAP数据!你也不能用!)% Y: j. B1 @! v F2 b
% T; H: m {5 b雷精灵:没错。但是MAP数据很小,我让MAP数据搬家,腾出空间来存放Frame图片的TILE。
7 N) f7 h& Q: \0 ^+ e
. N* a, @3 v/ m3 I+ o. p% x(众人:那么,MAP朝哪里存放?)
6 R/ D! P6 Y8 k( {4 X8 e) K* t! e3 C, |1 {) @: s W6 I, D
雷精灵:1号Char Base不是被占用了一部分么?把那没被占用的部分拿出来存放Map数据吧!
5 i- A3 m. U1 _: l0 Y1 v+ f6 C& a6 `8 C3 u N
(众人:(怀疑的目光)行么?)
5 ?- ^/ r0 G& n3 m7 q j6 d" ^3 X4 _3 o* u4 M
雷精灵:试试吧!
/ S; ~. K9 W- b
" z K: I- i8 k从主函数开始吧!- ^) @. N$ ^1 H. \/ s3 Q# [
2 X1 h9 P* y+ b0 f6 [/ L% |' Q
int AgbMain()
8 y7 F! m5 N- d& p7 _- ^6 a5 Z
$ k" W6 f$ V4 z2 S{
5 \. |" T; u9 _' q4 K
4 F( ^& }! {( OGsSetMode(MODE_0);
; V$ v# z; G, ]1 V0 s
8 y; D( x P) G8 y/ EGsInitBG(); 8 |( z! | s& g$ \7 l$ ~! M
$ v' y: X( Y( oGsLoadBgPal(BG_Palette,224);//与调色盘无关,不用管它。
) c8 N3 q9 z8 D; _: H# m' z8 c. v
# t' A* f' v0 U% E5 SGsLoadBgMap(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,看看它的地址……
( x8 P% p) H. X0 qGsLoadBgTile(BG_Tiles,sizeof(BG_Tiles),0);//还是一样,将256色背景图片的TILE数据装载到Char Base第0块。 ; T$ e: Z, R3 ]7 @
" Q; T6 C4 }: K$ dGsSetBg(0,Gs_BG_TEXT_SIZE_256x256,Gs_BG_COLOR256,0,15,0);//设置BG0:TEXT属性,大小是256*256,颜色是256色,其中MAP数据在BG的15号区域,TILE数据在BG的第0块,禁止该BG的马赛克特效。
; }( k7 Q! a, j: g- e& j3 d5 |2 w/ S; v. ~) Q" r
GsBgPriority(0,3); C8 n# ~2 L7 p: A
! ~" b, t) f+ s& q
GsSetBgState(0,ON);1 E6 v0 v: x: Y" ~
2 U8 J8 [3 m. a& u& P6 f" q
GsLoadBgPalEx(Frame_Palette,224,16);
5 U5 N' [' o v# u) x" P
2 g# A4 L4 E; |* o$ c" y& Z" _GsLoadBgMapSelectPal(Frame_Map,sizeof(Frame_Map),16,14);//那这里,明显就存放在16号得了!! K" a) @% u/ s8 ]0 m) K
0 ?# j+ o- ~, E0 O- o I9 bGsLoadBgTile(Frame_Tiles,sizeof(Frame_Tiles),3);//这个不需要变动函数,直接将16色图片的TILE数据装载到Char Base第3块就行了……
1 \% @2 Z3 w3 w2 C' m' N) YGsSetBg(1,Gs_BG_TEXT_SIZE_256x256,Gs_BG_COLOR16,3,16,0);//设置BG1:TEXT属性,大小是256*256,颜色是16色,其中MAP数据在BG的16号区域,TILE数据在BG的第1块,禁止该BG的马赛克特效。
! f8 Q' E- [% v E7 b& t! a
/ d' Q+ _4 w. tGsBgPriority(1,2);//设置BG1的优先级。次低。) S M3 E i. ?5 s$ i
T& I1 A6 T- I. l& I; ~
GsSetBgState(1,ON);//打开BG1。接着是文字层了……
# y5 ?. R* G8 z9 j0 T1 X, g3 O% G% Z) f3 F3 y
BGPaletteMem[241]=0x00ff;
8 W/ Z5 P' p3 s* A. G/ {2 T
( p. _8 F/ o' E: r% Q# h- d% ^8 ZGsSetBg(2,Gs_BG_TEXT_SIZE_256x256,Gs_BG_COLOR16,2,17,0);//设置BG2:TEXT属性,大小是256*256,颜色是16色,其中MAP数据在BG的17号区域,TILE数据在BG的第2块,禁止该BG的马赛克特效。, }! E& U; C$ ^ e+ c
, Q/ g; `9 d" P
GsTileTextInit(2,2,17,15);//那这里也是一样了。$ p5 l# g* o# a+ V: }; z
1 I# v: W; F' K/ \8 HGsTileTextOut(20,110,"我爱TGB,我爱烧录地带!",241);//……7 V7 K/ W! w' d* m# ^
3 w/ Q2 {+ D1 q% L2 C3 |GsBgPriority(2,0);//设置BG2的优先级。最高。
0 t9 j0 d- y8 l( {% [& r, }* d3 ]; t5 t" z7 Y
GsSetBgState(2,ON);//打开BG2。- O! u; Z* H# O2 N1 R
' L( V1 ]! s: h, a& o) HGsSetBgState(3,OFF);//关闭BG3。没用到的图层,最好把它关了!
! m( ~0 g1 p' L# m2 B" Q
& Y, Y. q" j+ U. d修改完了,试试看吧!* ~+ ?' d% W; Z, ^3 N o: p
) u% V) `1 a. ?( l' S7 L- J# e3 u/ X! y2 U8 D- q8 A

& a4 a" i# J, M3 u! f$ ?5 S _/ i4 j- \6 y: }( V8 J) A
5 d; g+ m* w& ]: f4 g) }+ U- A
什么!!!5 N, ]3 a3 Q/ W" T2 ^2 b$ H% f
+ i$ L% Z; H0 W( b哎呀呀……疏忽了!看来1号CharBase没剩下多少空间,把BG0的MAP载入的时候,还可以容纳,但是这个时候已经满了,BG1和BG1的MAP,跑到2号CharBase里面去了……$ `2 q& \& h, p: \" w7 K" U* Z
/ i% N. ]5 U2 |没戏了,MAP还是放回31、30和29号区域吧!好心没得好报!
: A/ T: q1 F( y% D( d# \* l. `$ y
( w: N; G U2 V1 l $ h9 f0 Z# c# s( r
" q% \# J" Q/ e3 z% h; i, ^! W
0 i* D" C; _6 r: F
终于行了……有够累!
1 [/ D/ M" U, I4 H6 d5 d- J+ p& t4 B g- x
从这个DEMO,我们能看出来,在作多层BG操作的时候,请多加留意!对于比较大的图片,你必须充分考虑到这图片的TILE是不是会超出一个CharBase所能容纳的范围!如果超过了,那么它的下一个CharBase就会被占用一部分,从而导致这个CharBase不能再容纳其他BG的TILE数据……当然,可以将其占用之后剩下来的空间作为MAP数据的存放空间,但是,你仍然需要注意剩下来的空间是不是足够放MAP数据……对于2层BG来说,基本上不会遇到这种情况,即使这两个BG都是256色。不可质疑这仅仅是“基本上”,无论如何你需要留心!相比之下3层BG,尤其是其中有一层BG是256色,这种情况出现的几率就会大大增加……如果是4层BG,仅仅是“如果”,那么这种情况出现的几率就是100%。这个时候请务必精打细算,仔细寻找可以容纳MAP数据,而又不会造成冲突的空间。你必须耐心的计算,到底哪里可以容纳MAP数据,到底哪里可以存放TILE。实在必要的时候,穷举法也不失为一种方法……
1 E- l- t3 j9 n' j8 _8 u" {, v/ _! i3 I0 N
但是,通过最近自己写的DEMO,我发现,某些图片,如果比较精细,颜色数较多,那么有一定几率会出现:这张图片的TILE数据甚至2个CharBase都不能容纳!这个时候千万不要进行大于2层BG的多层操作!否则,你会很沮丧的发现根本没地方容纳另一层BG的数据!这个时候你面临的就是,牺牲图片精细度和牺牲表现效果这两个抉择……不用我说,这两个抉择都是很痛苦的……
* ]; f+ f* K4 a
6 \' i7 _4 V V+ L" V1 V$ Q4 |嗯……差不多就这些了。到现在为止,如果你能完全理解上面几篇教程中的思想,那么你就掌握了GBA编程的三分之一。诸位不要说我罗嗦,再次强调一遍,TILE模式下的BG技术很重要……想进入GBA编程,请一定要掌握这些内容!
$ J: h, K1 x0 f( F% [% C6 C8 k' g K7 d/ Z/ j( q
那么,难点已经过去了,下一次我们来点轻松的——特效!在特效的配合下,BG就会变得多姿多彩!关于特效的调试,基本上不需要。你只需要通过VBA,看看效果是否符合你的想法就可以了。另外,我推荐的两种模拟器,在表现特效方面有一些明显的不同。具体地说,No$GBA表现马赛克特效的时候,是将这种特效作为水波特效处理的;表现FADE特效的时候BG和精灵不一样。由于在GBA机器上,特效的效果是和VBA一样的,而且VBA远比No$GBA普及,所以你就以VBA为准吧。下一次的内容:硬件特效!理论课内容:GBA的BG特效。/ v2 {' A* |/ A2 p
" m+ ~6 n( F( i6 t9 p
# b3 C$ f/ ?$ f$ U( s/ Q( n+ o" ?* w# B c7 m4 n. N5 @
9 |& D; R5 U! P. p: ? H
# b P. D; ~3 N' k: b. a
( e, x& u2 E. D$ R3 h) P$ S
P! X+ K% _, [9 i+ H- P+ e& U5 c" qGBA游戏编程菜鸟入门篇(六)——GBA的BG特效及硬件特效* N7 Z; A- q [
( u) D, |2 W. g. Y) H$ l8 Q' e2 R+ W特效,不是BG的专利,精灵也有特效。不过,我们暂时不讨论精灵特效。到了《GBA游戏编程中级进阶篇》的时候,我会详细讲解精灵,当然精灵特效也会讲解……
# P. J& p; v+ B3 s0 }! G" U4 k) ~% V
对于GBA来说,硬件直接支持的特效有4种:马赛克(Mosaic)特效、透明特效(Alpha)、窗口特效(Window)和明暗特效(Fade)。而通过相关算法,又能实现若干软件特效例如烟花特效、水波特效、颜色滚动特效之类的。其中有一个比较出名,就是伪3D特效。GBA超级大作、人世间最强大的GBA游戏、不到长城非好汉不玩这游戏就白活的经典、人类智慧最美丽的结晶、(后面省略形容词10000字)的游戏《黄金太阳》,将这种特效发挥的淋漓尽致、入木三分、(后面省略副词10000个)……. L0 Y6 I, y9 P: J8 w
L" i( L7 u' J0 m这一次,我们先看看BG的硬件特效。至于软件特效,因为这要涉及对TILE和MAP甚至调色盘的数据操作,而这种数据操作还不是一般的位操作或者循环就能完成的,所以我打算在《GBA游戏编程高级技术篇》中讲述……
" i& X, X/ v2 {
" U9 p3 s2 A+ N4 i d0 m7 X首先是马赛克特效。这种特效在现在的GBA游戏里面很常见。例如《口袋妖怪》,当口袋妖怪中毒之后,主角在大地图上行走的时候,每走4步画面就马赛克一下然后迅速恢复正常同时伴随着中毒掉血的效果音。再例如《恶魔城-晓月圆舞曲》中,游戏刚开始的时候主角来须苍真醒来,发现弥那小姐坐在他身边,阿鲁卡多站在右边。苍真醒来的过程,画面就是慢慢由模糊变成清晰的马赛克过程……还有几个GBA的Hgame,里面的马赛克特效实在是令人不爽……(众人:这小子有够无耻……). J9 ]' j; v+ e, [" P
1 b( }$ b& c1 k5 O& _( M马赛克特效的原理是什么呢?这不是很重要,但是了解一下也不坏。假如GBA硬件不支持马赛克特效,你知道它的原理后也能自己写出相应的算法,实现软件马赛克特效。
$ Q# ~# N* M) N# T3 M
& O7 @% c/ c* H$ ^8 i5 X1 H规定图片的左上角坐标为(0,0),向右横坐标增加,向下纵坐标增加。假如要做双轴3倍的马赛克特效,那么,横坐标和纵坐标均为3的倍数的坐标点例如(0,0)、(3,0)、(6,3)、(6,6)之类的坐标点,其左方和下方两个坐标点处的像素,均变成横坐标和纵坐标均为3的倍数的像素。我画个图表示一下:
& M5 I$ }2 D$ R! T) a2 X" f3 k
: k! h) p% U6 n6 V# |, X% \
" r+ }! i8 I5 _) i6 I F5 T- K' |3 s$ F) A" l- B

; N* H# P; ^+ I6 |1 I3 H9 {! B
I8 b2 k) o, H4 H* s5 D; G3 x! I# R Y0 W! Q; f) j" S9 [
涂有同一种颜色的像素,均变成它的右上角那个点的像素。这样,图片就形成了一块一块的明显的方块。
. Z' H: `4 f# | ]/ B; E# ~3 G- U! {: g% X9 m
那么,同样的,要做x轴4倍y轴5倍的马赛克特效,就是横坐标为4的倍数,纵坐标为5的倍数的点,像素值变成它的右上角那一点的像素值……可见,控制马赛克化面积的变量就是那个倍数。在GBA里面,横坐标和纵坐标的倍数都可以由你控制。
/ w. B5 `5 q0 C; G( c. G/ P. ]3 M- q) }3 b
那我们就做个DEMO看看效果吧!
; o' B* I* Z) s: x; W
) k1 {* m( M- W- ^5 h% l" w就拿写单层BG的那个例子吧。我先照抄程序,到需要修改的地方,我会加上注释。7 @4 z- y3 b( }, W
3 u. @% O, l: Z#include <GsGBA.h>
- T8 h& M: ^ h) R2 y5 F' y, t. X6 |5 j+ y! l7 T
#include "BG.map.c"
) s" X8 o2 r* ^& H4 w6 Y# G. J" u5 a( V ]3 w8 s3 g0 {7 F6 j
#include "BG.raw.c" 7 G/ W" C* q: N" h; F/ i% o$ E) v
% T: s& Q$ S8 w* j( |#include "BG.pal.c"
# b2 S# K I. n2 h6 O8 H* @" f
5 J8 F8 S6 j) S& [' tint x;int y;//这里定义了两个变量。这两个变量是后面对BG进行卷轴操作的偏移量。/ l, `7 p) Z) |2 p3 U
& s% u' w+ b: U7 n8 {
int mos=0;//这个变量是用于马赛克特效渐变的变量。% e/ j, A9 y* M- ]8 V3 m# {4 j
+ n+ E" x, x* D; Ovoid Vblank();5 J" |- d" c6 i3 k% P
& a" @5 x& Z, S! Jconst IntrFuncp IntrTable[14]={
4 h- d5 Y$ @/ ?. u1 B) F; T p9 p& B! k* [4 s
Vblank,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy& W" Z2 p8 r" d: x3 u/ d) ?7 |
5 H+ E* U Z5 f; t/ }};9 y$ @$ ]% Q$ [+ v
4 X4 E7 Q% A7 e3 x% t7 Z
int AgbMain() + U' U L4 Q( Z# L" q8 S. J0 [
{
+ C$ w! z; ~% [2 D2 }- I! D+ R" H
GsOpenIRQ();$ ?8 _! z$ y# Q# y1 v0 `
6 J5 a( v. }7 W2 ?+ u/ j7 w& s$ PGsSetMode(MODE_0); n8 U, u2 g- q1 R; j k6 Z! \
1 z- }. o- A2 J7 \; B
GsInitBG();
, p; H/ G" C9 H6 l% [ I
4 C* c) U: y8 ^ K5 m; T# hGsLoadBgPal16(BG_Palette,0);$ e# T; U9 {$ k! s2 D
) Q5 x3 }6 q0 G, ? kGsLoadBgMap(BG_Map,sizeof(BG_Map),31);- `; _% r) y. ~0 ^" S; e4 h
6 @9 g0 j, @5 U9 r( \2 b6 tGsLoadBgTile(BG_Tiles,sizeof(BG_Tiles),0);
! x2 H' Z6 O' j D( p9 P) {/ Z4 P" K x, U) U# u \
GsSetBg(0,Gs_BG_TEXT_SIZE_256x256,Gs_BG_COLOR16,0,31,1);//最后一个参数是控制马赛克特效的,打开它。
: `) G5 u, @: Q, t* U {$ @" K8 N" D. v0 T4 R9 q1 D
GsSetEffect(BLD_BG0_1ST,BLD_BG0_2ND);//对于硬件特效,设置完BG之后一定要记住打开特效支持!否则后面无论你怎么写,都是没有效果的。这次只针对BG0做特效,因此这个函数的两个参数均选择BG0。如果有两层或者两层以上BG做特效,相应的参数就要小修改一下。
0 A$ W0 K- t& j! Y$ a3 S+ [* q9 e3 p* P4 h. R5 v
GsBgPriority(0,0);
9 o! B+ K8 ?4 M: A2 i* V9 q4 S6 l$ ]6 U: c( e+ _
GsSetBgState(0,ON);
: ^& ?6 [8 `( _) U2 [+ J+ p) O) L+ @2 C( \, a4 p; I/ |
while(1)
" Z- z( W$ \( s1 N. U6 F8 z2 a& B! ?5 A
{# X* ?6 Z4 G1 u. o6 o
9 @: q# P u# d* |4 u( W7 [6 z; t$ N5 BGsWaitSync(); 3 ]' B7 w) G E; r
& G: z' c- C# L! r
}
3 |% l F6 Z4 p0 X" k; @
* f, z. y+ A, S# ireturn 0;
* H+ ^( B0 ]! N+ f7 I5 S+ Q& O" V
}
$ _- V" m/ X6 w( X( o$ a) x ^; \( R3 A# e
void Vblank()
% k7 t; c# Q6 @4 _6 L2 l6 Z- i/ R* t* V0 Z7 H& _! `( \4 Y% x- p. q
{. y$ [: t0 d: k5 ]. T; C2 g, h; I( s
. T4 e+ H: @8 y5 {3 B
KeyRead();
- H3 _( e$ q- a
1 [+ V6 m5 A( t; U% Bif (Cont & L_KEY)x--; 1 C, @7 W+ C) J: t
; j4 K: U4 L- O# A: I2 g' L
if (Cont & R_KEY)x++; 5 g; i7 ?4 K! i1 U0 l( |, i
, b3 V& A4 j3 O+ {. w9 }if (Cont & U_KEY)y--;
+ B) Z8 i( L* S0 O+ ]. M( z
) U6 U. O5 x7 ]' i8 Z1 _if (Cont & D_KEY)y++;% c- F6 D. t3 E; ?+ N
% ^8 U4 ?- u. N% @5 d4 a4 `. {4 k1 e
GsBgOffset(0,x,y);
# W$ `% n8 o7 M0 f; q# ]6 H4 w* Y$ t/ b" v4 g2 s
if (Trg & L_BUTTON)mos++;//当按下L键的时候,马赛克变量自增。这里没有用连续按键。因为连续按键的话,马赛克变化太快…… L, ~: H! [, z
. A' S- h& b4 f$ c" Eif (Trg & R_BUTTON)mos--; % r9 J% i, u: g; O M8 ]
, Z: x$ ~2 Y. J! q& NGsSetMosaic(mos,mos,0,0);//同样,这里要对键盘操纵的马赛克变量做处理。我们这次的DEMO没有用到精灵,所以后面两个参数均为0。
, b% ~/ g) }- g; C- N/ p' ]+ i& [/ q+ m- S( T( ]3 b
}. B0 z) i7 t2 E- x# D# k
4 t" z$ S5 N* C- E这是效果。: u, @. e0 ?" r1 k
1 d G1 o# I# a; A5 {
2 d2 j; A& P9 P: d! T& U2 Y; x
" H$ O: |# L: f. Y+ \* `/ k1 {
6 p q0 T; d, F% _" ]; F, s, p( U
/ [2 w. S9 B/ Z' e注意,马赛克特效属于破坏性特效操作,就是说,如果进行马赛克特效之前不对图片的数据进行备份的话,就不能恢复了!这一点,相信大多数朋友深有感触……例如一张H图片,重要部位被马赛克处理了,那么想恢复原状就是不可能的。(众人:(鄙视的目光)只有你才深有感触吧……)幸运的是在GBA里面是有备份的,而且这个备份操作是自动的,不用我们操心…… x9 m( c* T3 @, X3 w, }8 M
$ W O( S3 V% c有关马赛克特效暂时说这么多。其实灵活运用马赛克特效可以作出一些很出色的软件特效,例如水波特效。有关这方面的内容需要涉及物理方面的牛顿定律以及数学方面的正弦函数和随机函数,应该说比较深奥。到了《GBA游戏编程高级技术篇》里我会简单介绍这些软件特效的算法。
4 x" V" p, U1 `; B& J; O8 e: H: U# X/ E6 f
接着是FADE特效。这在GBA游戏里面也是很常见。例如游戏开头的LOGO。我本人就很喜欢在游戏刚开始的时候显示我的LOGO,然后用FADE特效将LOGO变黑,然后慢慢变亮,停顿几秒钟之后再慢慢变黑……% C' S9 N& r8 T4 m$ c
4 Z6 C3 R5 t9 Y9 c- \: \6 J
来一个DEMO吧!还是用上面那个例子。
" X, l) \6 H+ V: ~# o1 a1 ]% `) `5 ^' E, J# s
#include <GsGBA.h> , U; ]# [( A; Y- _
7 @ J: z; B) E" V# u3 t9 L#include "BG.map.c" 2 A. V! ~8 ?& a9 ^3 O3 y7 @8 ?
* m1 k- k$ m, x2 F/ Z+ }
#include "BG.raw.c"
7 b- m! m8 x" ~# q) ?9 V
9 p9 v3 H" y. e; F9 u#include "BG.pal.c" 9 m0 ?' R1 P- ?. V- C _
4 i5 v0 J% v6 [& Z G' P Dint x;int y;//这里定义了两个变量。这两个变量是后面对BG进行卷轴操作的偏移量。) U5 a. S. J6 c5 R
9 v% p3 g1 ~3 h' n7 s/ E K+ ~2 Yint brightness=0;//这个变量是用于FADE特效渐变的变量。0 x& q/ x4 y }: `* w, `3 V4 E. K
& p* _1 R* x6 [+ Q& h, r$ n
void Vblank();
7 W- j$ a; |& F6 L# `" u( s" o7 j/ F2 |& i/ E% K. u' w( W
const IntrFuncp IntrTable[14]={: q6 i4 R: T6 j) o$ v# K+ H1 v" U
- R7 ~0 Q' |$ N G, KVblank,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy- V1 }5 p$ T- z* R, W
/ c& c) E. m- x$ k0 l, a0 m8 W
};/ v9 b: \, `6 `& t
: ^; p) u: d; d1 k& x: \! o
int AgbMain()
: `, F" D' ~1 H{4 P: u$ _: x' I. G1 A' @1 @
4 t8 ]# h: ]& n: D4 a7 \GsOpenIRQ();" a& O* c% V% P8 M, O
5 `) _% v+ g# d- j# v( U; FGsSetMode(MODE_0); S7 Y3 p# K3 p: v
( i3 S1 Y1 X3 a |8 c4 L8 m0 ], c9 V
GsInitBG(); ! e) ~) }) w: M% V# E
: S! \- Y1 T. U/ D, V, k8 sGsLoadBgPal16(BG_Palette,0);
5 s1 \7 c2 O8 F2 h* p# c, R3 p+ v8 X8 l% h& Z# }( P) ~4 `. r
GsLoadBgMap(BG_Map,sizeof(BG_Map),31);
, J. M5 X4 R& y; z H5 d
1 \5 d, L( a% y# {! ]GsLoadBgTile(BG_Tiles,sizeof(BG_Tiles),0);7 k/ z+ d+ L2 U4 w. J
+ w! y( E* o; K* O0 KGsSetBg(0,Gs_BG_TEXT_SIZE_256x256,Gs_BG_COLOR16,0,31,0);//最后一个参数是控制马赛克特效的,这次不需要马赛克了,关闭它。
j8 g N8 w8 J' |( e. e( _5 R) A
7 I7 t7 A ?! J( D8 |GsSetEffect(BLD_BG0_1ST,BLD_BG0_2ND);//同样,设置完BG之后一定要记住打开特效支持!这次还是只针对BG0做特效,因此这个函数的两个参数均选择BG0。
" V5 }2 N3 N& K2 B/ y3 E
, M5 A S% h$ a' X/ {GsBgPriority(0,0);
! `& h: ]! g' ^% F) }1 c! h
: }! J7 h+ m2 K2 V2 K4 D2 UGsSetBgState(0,ON); 8 a3 |$ P" D* }
8 n/ v# w/ w" Q& W
while(1)
" ^: M8 l1 F+ T& u1 s
! W% u" |4 @9 }$ b{
- I$ [9 K' T; o3 p" J3 k( W. \3 Q! J
GsWaitSync();
8 X+ L1 X' U+ H1 R9 y! R
) K1 ~+ c9 l p7 N# E} : o* r- D5 l, m) s! W
# j0 L& Q# l( _- {7 L
return 0;
$ r) j/ T# @8 E! a3 O# Y
- z- e+ U- |: `/ V* p+ c b7 j6 ]}
# m3 F- |1 c) n* m3 I- m9 M- T6 l. K2 |- r4 z; s: Y, c
void Vblank()
; Q0 |# m; ^) y7 D: v1 h" h/ Q* Z% D0 {9 z
{
: f/ E8 n- O% K) M" X6 ]# X4 Z G; n0 s# }, t$ o
KeyRead();
/ {% i+ w8 M9 q1 ~7 ~2 d3 J; r, X* K& J
if (Cont & L_KEY)x--;
) M' Q F" i, _* G" e. Y3 z, ?" N; Z
if (Cont & R_KEY)x++;
4 E6 {9 Q |3 |% G S
+ y$ I" p+ I. h, uif (Cont & U_KEY)y--;
7 l0 b3 y' L X% s2 D, v/ W- L* b! L, M
if (Cont & D_KEY)y++;
. \: Q) [6 m; q5 j) i+ K" Y! a- i9 t: z# w/ o! i, |2 o# g
GsBgOffset(0,x,y);
# Y, j' e. [" ]0 D3 F
1 Y! ]: e! }% m! K$ ^if (Trg & L_BUTTON)brightness++;//当按下L键的时候,亮度变量自增。这里没有用连续按键。也是因为连续按键的话,亮度变化太快…… K/ i9 ^# a4 m ?6 _
2 b, c# e2 r# o5 u* M" H
if (Trg & R_BUTTON)brightness--;
! Z! C; B$ L3 X1 f, h9 d4 K+ j+ D6 y& N! P; r
GsSetBright(brightness);//这里就直接控制亮度了。要注意的是,帮助文件中规定的亮度参数是s16类型,也就是允许负数。但是即使你把这个变量定义为无符号变量程序照样正确。原因很简单我就不说了。另外帮助文件中规定参数范围为-15~+15,其实超过这个范围也无所谓……你可以试一试!; Y$ ]0 S" V. s3 [0 G- ]2 i
( [( {/ O3 Q: }$ O6 B}
( ^( w2 R, V+ {$ }: b5 x# b! [. a/ t
效果图:
8 u6 H8 S! J& ?0 J& x+ l: `+ W4 w
3 d' T8 D% o! e6 ]' y: O" E9 X" v, M1 H+ P: Q3 N( r# o4 r
* A2 T1 p l6 C1 b' K
9 A5 U' j' C- F+ }
' \1 B, u7 b% m8 e! ^( f. [FADE特效也是破坏性特效操作,不过你仍然不用操心……嘿嘿……$ M5 A, e ~& Q" t
" V9 F/ H4 ^7 E f5 |: |& j" z* l
利用FADE特效,可以很方便的做出黑天特效。GBA超级大作、人世间最强大的GBA游戏、不到长城非好汉不玩这游戏就白活的经典、人类智慧最美丽的结晶、(后面省略形容词10000字)的游戏《黄金太阳2-失落的时代》中,有一个这样的剧情:主角们旅行到一个狼人村庄,刚走近村庄的时候,天暗了下来,然后迅速天黑了。(很令我奇怪,无论我走了多长时间总是到了村子口天就黑,即使刚从客栈里面爬起来……这几位兄弟时间掐的还真准!)这里的特效就是FADE特效了……当然,也可以做出刺眼的光照特效。仅次于GBA超级大作、人世间最强大的GBA游戏、不到长城非好汉不玩这游戏就白活的经典、人类智慧最美丽的结晶、(后面省略形容词10000字)的游戏《黄金太阳》的游戏《口袋妖怪》中,神兽古拉顿觉醒之后,整个芳缘地区开始出现烈日天气。那种刺眼的阳光,就是FADE特效……
) ]/ K6 K8 ^; M7 V
3 F' f. i9 P. `' U. S2 W; V然后是窗口特效。顾名思义,窗口特效就是BG上面出现一个区域,限制了BG的显示范围。我孤陋寡闻,玩过的GBA游戏很少,迄今为止只发现了一个游戏使用了BG窗口特效——《RPG制造》。游戏制作中有个选项,就是游戏结束后出现STAFF表,是以什么方式出现。有一个选项是背景图片被挤到屏幕左方,右方空出来的区域,显示STAFF。这个显示过程就是窗口特效。; E: V: {+ C c0 B/ t' y$ B
/ k" b, j9 D6 U1 x/ _9 }
GBA硬件共有三个窗口可用:窗口0、窗口1和OBJ窗口。其中窗口0和窗口1效果相同,呈现为屏幕上的一块矩形区域。OBJ窗口可以算是OBJ的一种特效,它只能配合OBJ使用。一旦OBJ使用了OBJ窗口,其显示就会变成一个轮廓。关于精灵窗口特效,最经典的例子就是GBA超级大作、人世间最强大的GBA游戏、不到长城非好汉不玩这游戏就白活的经典、人类智慧最美丽的结晶、(后面省略形容词10000字)的游戏《黄金太阳》中,角色们在大地图上使用“Psynergy”(好像是被翻译为“精神力”的吧?),其身体就变成了一个轮廓……还有,在阴影里面使用Psynergy“阴影”,身体也是变成轮廓,而且很明显……
3 p1 m. Z& T$ Z: x: m0 \, m8 C
7 o2 G0 e7 S* u$ i# z: c我们也来一个窗口特效的DEMO。
" T! r6 p& ?' d" x6 U1 E$ T: J" N2 K1 x8 y# T
#include <GsGBA.h>
0 ]" ^6 G5 ]9 B! b7 W Q) k' v4 K) j
#include "BG.map.c"
! v T V7 q5 N# S! U
2 [* `1 ^5 F1 d3 j6 l- n#include "BG.raw.c" + |3 g6 s+ n3 {
8 a* L1 q9 Z1 z( w#include "BG.pal.c"
- @" \' f. e# v: ?
% S* r y1 ^/ U/ j) a$ b& e% ]#include "Blue.map.c"
8 y% k& q. ]8 ~3 v8 O5 A" X b
8 j3 v- x8 a" zint winx=0;/ T0 Y- |! Z+ r- `7 U
( |( N9 L( x; Y* E$ F, ?int winy=0;//这2个变量是用于控制窗口大小的变量。
* @+ A! l* O% I j r+ V
: m, `; M0 {# T3 l( j1 Z+ ^void Vblank();1 W/ b1 u E& c! A' _9 x' I$ M; f) K
2 Z! ^9 e$ ?8 v$ ^1 x
const IntrFuncp IntrTable[14]={5 N$ B7 y- T+ k+ O, k
7 j, P# x8 U. r9 @1 n- ~: yVblank,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy,dummy
y) r3 u) W7 ]2 A; c2 c+ F! {' {+ M2 r% b' c
};
9 |7 e8 ~; k; ~2 s4 t' |. g$ ^$ N T6 g* n, x
int AgbMain() # V: h$ L, \; [( D! r; R& b- {/ i
{* t0 y3 m' E2 _+ @& M2 c( h
+ U* X0 @6 O9 EGsOpenIRQ();6 F* b* C# R- u0 g/ O6 U/ B
% M8 G% e' q9 g) r
GsSetMode(MODE_0);
: `8 s' {# |& l! }
% n9 E2 w5 `4 AGsInitBG(); & I, E; i: W5 g+ q4 _/ e
- y+ o* ^$ y* N, U' I& KGsLoadBgPal16(BG_Palette,0);
! M; u$ U; a x( x( W; o3 h6 l; o2 z/ \9 @% u% r
GsLoadBgMap(BG_Map,sizeof(BG_Map),31);; ^, Q! \/ [: q& [0 z
^% i7 g1 P' V: vGsLoadBgTile(BG_Tiles,sizeof(BG_Tiles),0);
1 t0 H/ F3 Z' P3 i7 C% d" X
# A8 J3 I5 A& Q5 sGsSetBg(0,Gs_BG_TEXT_SIZE_256x256,Gs_BG_COLOR16,0,31,0);! B! K$ [# a" _" H& z; v
% a0 _* D' Z2 c( tGsBgPriority(0,0); + W/ N9 K& i& p. }
' e7 C6 D6 E/ C/ l! }
GsSetBgState(0,ON); ; c& W1 Q) t5 J8 W
# O4 C3 |3 I k1 iGsSetEffect(BLD_BG0_1ST,BLD_BG0_2ND);//同样。5 y- s4 H5 T$ ~4 c; U, k$ `
' g7 ^2 G' v" O6 ^9 o
GsSetWinState(0,1);- F+ L; M( \+ a
* P: w# g* ]1 N3 D% g& wGsSetWinState(1,0);//设置窗口状态。我们没有用到窗口1,只用到了窗口0,所以激活窗口0,关闭窗口1就可以了。* s0 w7 b! b+ z3 D! N8 h
+ r2 i d4 O* x2 U a9 Xwhile(1) 1 w: J/ [! G. H+ w
# {( [# i0 K2 q) k( ?+ \
{3 r1 X7 g7 g- P( p$ `! A
. w1 Y4 M; q I3 {1 H& H
GsWaitSync(); . v- O6 K( H1 {9 O2 F
1 ^" n2 U& ~; S! d& r}
" w/ V+ s+ g0 g7 w6 X, S% x9 n) d- D- n& p5 ]. Z' g* j' ]
return 0;
: U2 k9 Y# E5 d5 q7 Y" ?" {' z& c' a3 J3 v6 Q
}
0 B- y2 y$ Y# r" F$ _3 Z$ y/ w" e/ @# |+ X: ?+ Y& t
void Vblank()
. u4 Z- j8 h; U- f. b8 k7 w. p2 n9 i, f; F; M) S7 e
{
7 b$ M* |: r! o# l+ U* z2 y9 z$ `, R9 m7 G
KeyRead();- @. z; U4 d5 R9 S. B" E
3 O6 s: E! P& T$ u# Y7 e; m" Z
if (Cont & L_BUTTON){winx++;winy++;}//这会有什么效果,你能猜出来吗?
- P) m; c5 F3 Y# }% }& J, u
* M7 M' r+ o* B; ]if (Cont & R_BUTTON){winx--;winy--;} $ Q( J0 l- z& P- k7 a
8 K: X1 w* R# u7 rGsCreateWin(0,120-winx,80-winy,120+winx,80+winy,WIN_BG1_ON,WIN_ALL_ON,ON,ON);//创建窗口。使用窗口0,大小由我*纵,窗口里面显示BG1,窗口外面全部显示,允许窗口特效。BG1我们没有用到,所以就是“空白”。- }! n) h" ]) L* o, |' N
; ^2 ]8 N4 M8 G4 f& X}* U! j: _+ p( Q( P- d
* T1 K% D( L' Y这是效果。, F f* B0 p" E) Q
1 x u2 x. X i! Y
$ h O7 t# o9 B+ R. v8 W7 v
5 a" Y" d& ]% j& F' c/ ]
* F6 T6 F5 i% u* f: \$ k4 Q: r. F$ p; p& k% K5 k
窗口特效不是破坏性操作,从宏观上看,其实窗口就是一个“虚拟”的BG,这个BG盖住了真实BG的一部分……形成了像上面的那种效果。
: ^2 F8 o& B7 c A
; @# a/ [) z( X, P9 ?0 v在GBA中,存在两个窗口,就是窗口0和窗口1。理论上讲这两个窗口都可以由你操纵。不过,或许是我的分析有误吧,我只能操纵窗口0,窗口1操纵的时候,总是出现异常的状况,所以,有关窗口1的资料,我暂时无法提供……召唤达人中……如果你要使用窗口1,请慎重!9 t- |3 g1 ?* I! t
! _& Q; N0 p2 u最后,剩下Alpha特效了。大家都知道“三原色”——红绿蓝。这三种颜色通过不同的比例混合,可以做出各种各样的颜色来。GBA硬件可以完成这种颜色混合。这种特效,就叫做Alpha特效。( x/ i4 _8 k1 g8 |+ l
2 Y0 u- C1 _9 z# t X, t这种特效在GBA游戏里面不怎么常见,我只见过一次——还是在GBA超级大作、人世间最强大的GBA游戏、不到长城非好汉不玩这游戏就白活的经典、人类智慧最美丽的结晶、(后面省略形容词10000字)的游戏《黄金太阳》中,片头黄金太阳升起的时候,背景的阿鲁法山在阳光的照射下显得颇为壮观……阳光穿过浓厚的云层射下来,形成美丽的光环。这个光环,就是活用Alpha特效做出来的软件特效——光晕特效。我个人认为这是所有游戏里面最令我陶醉的一幕画面!这个特效,把这一景观做绝了!
- X0 N7 f: X3 l7 v* v7 [
) f+ y4 s9 u: M6 D$ p W _ a3 X; k5 l) V& C

$ R, N( }) k' D% S# ^( M. n& J- A
1 f& A9 Z$ E- x5 g5 ?0 ]4 F( E1 v- d7 U' n8 D; ]. D$ y
关于这个硬件特效,我不想写例子了,我觉得,大家参考samples文件夹里面的DEMO,自己试着写一个就可以了。还是那一句话,用的好的话,这个特效是很炫的!
; E0 u/ I1 V! Y
$ }8 Q3 D: [5 `到现在为止,有关BG的内容就全部讲完了。不过,这并不是说BG就这点东西,更深更好用的函数还有呢!下一次,将是《GBA游戏编程菜鸟入门篇》的最后一次教程了。学完这最后一次教程,你就不是菜鸟了!因此在中级教程里面的例子,我不加注释的部分你应该能够看懂。如果你还是看不懂的话我劝你还应该再从头看一遍。下一次的教程,GBA上面的音乐播放!学会这一部分,配合特效和多层BG,做出来的效果不用我多说什么吧……嗯……还有一点小小的事情:如果你有点五音不全或者对音乐的鉴赏能力很差,那么你就不要学下一次的教程了。因为下一次的教程要涉及自己写MOD,这是和写MID非常类似的工作,如果你连MID都不会写,那么MOD你也未必能写出来……
8 A* R* q, E" o- H& R B3 N7 I) t) g) H1 D
GBA游戏编程菜鸟入门篇(七)——GBA音乐播放 $ T' y$ t$ z( E: f9 A2 n8 @
1 g" g- U/ |* U7 z1 b. T' ?
0 A3 K5 }% D' Q$ A' GGBA上面一共有6个声道,分别为SOUND1~4,和DirectSound A/B。这6个声道硬件上到底是怎样播放声音的,这个问题实在是复杂,官方开发包里面的帮助文件和寄存器映射表我基本上是完全看不明白。不过,GsLIB中提供了一个非常好的工具——MOD播放器!这个工具,使得我们可以非常容易的在GBA上面播放声音,无论是背景音乐还是效果音或者真人语音……不过,会有一点点的失真。% T+ @( T3 r% {& @
% U" F( u# V# b. W; }9 h在GBA上面播放音乐有两种途径,第一种是使用GsLIB2.6 ARMSDT版本提供的工具——ModPlayer——来播放MOD;第二种就是只有GNUPRO版本的GsLIB才能实现的,使用MP2K播放MID。ModPlayer有种种缺点,但是由于使用简单,我们还暂时不能将它抛弃;而MP2K才是终极之道,官方游戏,大多数都是使用MP2K的……如果你掌握了ARMSDT版本的GsLIB的使用,那么我强烈建议你转向GNUPRO版本!因为两种版本中的函数基本上都是一模一样,相信你会很快习惯!
2 e( L5 U2 z$ q- f: S
( _( e3 ^$ P6 s/ R+ O首先简单解释一下所谓MOD。MOD是一种音乐格式,和MID类似,只不过自带采样数据。帮助文件里面就是这样解释的。详细的资料,我也不知道,你要是有兴趣,可以到网上找一下相关的资料。在我们这里,了解这么多就够用了。
/ ?% K1 e/ o( W5 p* d: W# D+ f& y* K0 f! `5 q) ?
既然MOD自带采样数据,那么MOD实际上就可以用于播放WAV,只要你的采样数据是你所需要的WAV……因此,如果你想在游戏中加入真人语音或者效果音,只需要录制好你需要的声音成WAV格式,然后在ModPlug里面导入你做好的WAV成采样数据,然后……
, W+ T G* l. Y4 L+ g+ i) c1 i; Y- a- p6 o) {: N: |
首先说明一下,制作MOD的软件应该很多,但是我只找到这一个;不过这个软件也的确强,虽然做出来的MOD有点失真,但是这很可能是由于我对软件生疏,制作的时候造成的……不能不说这软件真的很好……由于我也是刚刚使用这个软件,操作的时候出现的错误在所难免,请各位见谅,雷精灵也就只能给各位提供这些资料了……当然,如果您很擅长使用这个软件,那么还请您多多赐教!
0 S- m# M# b- |' P9 z. n3 f3 [: F- D
哼哼……我们来实际操作一下!
7 d% {8 H( c- ?) \
: d. J& ]) v- _$ t# S首先是ModPlug的界面……
; K% d9 @' M6 w
! G2 f: \" |7 o) {1 t$ Q+ l, S! S+ `8 n9 H
这个时候你可以用它打开一个MOD文件,熟悉一下每个部分的功能和作用……关于MOD文件,samples文件夹里面有两首很好听的MOD……& b7 D7 m( _8 p! W5 q# w% t
) n) \( \. B. @自己录制一点语音吧!我从游戏里面截取了一点声音并转成了WAV格式。这是PS《生化危机3》钟楼里面的八音盒音乐,很爽……不过要注意一下,转成的WAV,最好不要是立体声,我不敢保证立体声的采样数据能不能做出MOD来……5 O7 {. ?6 C% q3 y+ ]1 o
0 T$ q* m) e+ ]$ j" a, ~+ s点击这里下载《生化危机3》钟楼八音盒WAV音乐:http://www.tgb.name/index.php?act=Attach&type=post&id=516
3 i2 ]2 w( v6 k
( D) L" }( C2 v; B9 w8 }5 i然后打开ModPlug,新建一个文件。$ ]( H& Y- c$ h; Y" R7 n% b c
$ v" N+ Y1 u' h# F+ }1 f
然后主窗口里面出现了5个选项卡,分别是General、Patterns、Samples、Instruments和Comment。大家的英语一定都比我强,我就不翻译了……
) H4 y; L5 `. y9 K% j0 y% H6 A6 }4 K" O" O8 `
点击General选项卡使之前置,最右边是声音的幅值显示器。这个东西大家一定见过!那两路显示,代表声音的两个声道。 q. H1 I8 ~# ?2 g- C1 d6 u* o
3 o U" }- H* ?; R( L
关键是它的旁边那个“Change”按钮。点击,弹出来一个窗口。把Channel数改成4。这样会减小MOD的体积,而且由于是在GBA上使用,4个Channel也完全足够了。5 |2 y9 x. Q# ]
0 o0 ^0 v$ e8 i+ H8 |. O
然后点击Samples选项卡,这个时候采样数据前置。 d) b: `, x( O4 ^8 R# a/ H
& T) A8 E0 K1 P$ G* J' O
9 G! |8 f- B9 m- ^. W% g

4 V e4 K2 t7 [0 P! v6 C. Q8 ` |5 R+ m$ v1 Z
直接导入采样数据,也就是你准备好的WAV文件。导入完成之后下面的波形窗口就会出现采样数据的波形。点击播放按钮会播放采样数据。
8 @( n5 J* L3 T# w4 T7 T- n# K
! x7 R a+ C+ E下一个步骤非常关键!请务必照着教程做!
, X8 j/ c5 O% _5 A
e. r+ _: m0 g/ c# Y
7 T2 y) T4 } y7 I
% j& l+ \: E3 ~& }* n
- c6 _0 R+ e: D) A2 I6 }
! e9 _' M* r2 l3 O点击两次那个“Downsample”按钮。这个时候下面的波形会有些许的变化。
$ m- ?4 A' g: K: F
' L, f+ [- d8 R @- Q然后点击Patterns选项卡使之前置。准备写音符。! u) h1 x, @& P q/ J5 x5 n% t- O
' {7 `- H1 K l6 P# r! i1 h- C双击第一个Channel的第一个音符,弹出音符属性对话框。音符选择“C5”,采样数据选择第一个,就是你刚刚导入的采样数据。由于我导入的采样数据比较长,我新建了一个Pattern,这样可以有效的防止声音被掐掉尾巴。9 {! J$ a4 }7 @; ]
) k& r2 g* Y) L! E# T3 O, B/ h b* g
然后直接另存为MOD格式。' f: n* m4 t6 ?. S
$ ?7 Q3 h6 [* x) t/ J: W$ W
关闭,再打开,听一下效果吧!虽然音调有点低沉,速度变慢,有点失真,不过效果还是颇不错的……如果你想做成立体声,就把另外几个Channel也填上音符C5。6 Y$ D2 m* {, a( n9 ~
: l, G, N8 Q; @+ J, q如果你对这个失真很在意,那么我建议你先用变声软件处理一下你的采样数据,然后再导入……8 W2 A9 L/ [9 Y; Q
$ [ U) a- {; ]' }
那么,MOD有了,开始写程序吧!: _1 v- `% _/ k2 p: L3 g7 I
4 q* h+ C- T! E& f! `(众人:嘿嘿……这还用得着你写?光看帮助文件和samples里面的例子就能够写出来了……)" N/ E! |$ C5 Z: `4 f+ [0 g
& A& M. _ J8 }雷精灵:(尴尬)啊……那……好吧……我就把我写的DEMO放上去得了!$ j( j# a6 ~* M$ ]" I, c
0 E' r# V3 p+ Y- {4 R
点击这里下载DEMO:http://www.tgb.name/index.php?act=Attach&type=post&id=518
1 g# f4 {& _4 ?
% c% w% |/ ^% ?& p) ^8 L就像这样,GBA可以播放音乐了。只要更换你的采样数据,就可以做出各种声音的MOD。如果你导入两个或者更多采样,写音符的时候在其他Channel使用别的采样,那么,效果就不用我说吧!如果在其他Channel使用相同的采样数据,但是有点延迟,那么就可以做出回声效果……只要你能想得到,就不怕你做不到! z, m0 l) O) T9 w# N
* s; x) J/ Q# [7 h7 _0 b9 n8 _
同样,如果直接使用现成的MOD音乐,就可以当作是背景音乐;录制效果音作为采样,那么做出来的就可以当作是效果音;录制语音,那么就可以实现播放真人语音……有没有人放一首歌上去?诸位不要笑,完全可能!
+ o, S, b5 F0 L! ^* m2 t9 L" u+ T8 N+ m* w3 X; L
到这里,初级教程就完全结束了。现在的你,已经不是菜鸟了!毕竟,能够自由的操纵BG,随心所欲的制造特效,还顺带着加上背景音乐,已经很了不起了!下一次教程,我们就涉足更深的内容——精灵使用!精灵技术是游戏中的一项革命技术,掌握了精灵使用,你就掌握了GBA游戏编程的另外三分之一!
( F6 r# r7 N/ i- R/ y3 {- g- |# W5 a$ X6 x: N
GBA游戏编程菜鸟入门篇(特别篇)——制作一个MINI游戏
5 p8 M: z; A& T8 N$ ^
( Q+ W0 a# B4 B3 b0 V1 [# R+ K0 j S初级教程中,我们掌握了TILE模式下BG的简单内容,学会了使用多层BG,能够使用简单的硬件特效,也可以进行音乐播放。这次的特别篇里面,我们将尽量多的使用我们学过的所有知识,制作一个MINI的游戏!
" j+ ~) k2 r2 L2 C# ~2 ^- I7 T* B1 k$ c `
一般的游戏中,哪怕是很小的游戏,“精灵”技术也是必不可少的。不过,我们还没有学到精灵技术,还暂时不能使用精灵。但是,极其简单的游戏中,如果活动物体很少,并且不需要进行大幅度的变动,那么也可以用BG代替精灵。这次我们要做的游戏中,就用BG代替精灵。. M; S2 r+ _6 \
" D, a+ Z- ]+ L ^
做个什么游戏呢?HGAME怎么样?
( M ?7 I, t- f; a# G* w+ ?5 x- h* |1 i9 u! s" ?
(众人:(怒)你个色狼!)
/ N6 |# g; I3 E: q2 w0 ]: d M
; W! a. d, p3 r: ~7 L" q雷精灵:大家不要误会呀!HGAME一般注重情节和CG,活动物体和精灵一般不多。所以,这个时候选择HGAME,应该是合适的……
, ]' R1 R; ?; V5 H$ @' N2 U" j- n5 L- D" x* v
我最近在网上看到了一个挺有趣的HFLASH,大概的情节就是,游戏随机出现一个数字,然后下面的秒表计时。如果你在恰当的时间按下按钮,使当前秒表显示的数字和给定的数字相同,那么游戏里面的女孩就脱一件衣服……直到你把女孩脱光……相反,如果你没有准确的在恰当的时机按下按钮,那么女孩就穿一件衣服……若干次失败后,就GAME OVER啦……
" |) ~: K/ y0 R( b' J, \, V5 d
+ I- [, Y; E2 d7 V [$ j0 O如果要把这个游戏在GBA上实现,应该怎么做呢?
( t( y0 I4 y$ C, u V" H
* m: K7 K6 o# L; d1.随机出现的数字,可以用随机函数确定,数字,以文字的方式出现,可以用MODE0下面的文字函数实现。不过,我这里最好还是先说一下,关于这个数字,最好还是用精灵的形式实现。因为用精灵的话,你可以将文字做得很漂亮……不仅如此,精灵实在是太灵活了!用好了精灵,效果真的会相当出色的……
: U$ ]- u% u7 d# Z/ F0 J* i. ^3 c6 k; R; }
2.秒表,因为频率是固定的,所以可以用中断实现。VBLANK中断,它的频率就是固定的,非常接近60Hz。秒表显示的文字,当然也是用文字函数。
: _9 s' A9 U% N# A, N, B- t- T( P& ~0 f" h
3.按键响应,也要使用中断。检测到按键的时候,暂停秒表运行,判断两个数值,然后分支。
M g; l8 Z9 u1 W- ?; t, c
' S w' e5 |5 I3 ?+ b) z* l4.女孩的图片作为背景,根据分支确定背景的显示和替换。- P' d( J/ _5 _& w
; K0 s5 t) h3 y; F3 J* A嗯……大体就这么多了。开始吧!
$ q! ?, V, x9 D: @: i x2 B$ v# K) D+ c: V& j& _
首先准备图片。这个游戏的所有资源,在教程的最后我均提供压缩包下载。: D7 E; X( N0 K+ j" C' B
: l. W( X3 @4 s( E& i, V) YCG,BMP格式,256*256像素,256色。因为处于BG最低层,不设透明色。* I, P& M$ ~! \- G2 A) b/ G" N
+ N" `4 \. P2 H
文字层1,用于显示随机出现的数字,还有一些对话什么的。调色盘使用16色。
" @6 R. H- p) f, o p/ P; `$ t6 N6 A9 C/ t# s0 @% k/ g: d0 u
文字层2,专门用于显示秒表数字。调色盘也是使用16色。* u" C7 ^4 ~! ]9 _
/ g; @2 N( S- c$ p. G$ q9 u
为了出现比较好的效果,加上游戏片头。也算是为自己宣传了吧!游戏片头使用256色图片,256*256像素,使用FADE特效。- ]' B5 e* |+ Z9 Y/ ^% G. `( l
" W0 P1 x/ U* p7 A8 { b游戏标题也加上吧!使用马赛克特效。
. k: U# |7 w: s5 s" z9 I; E8 N& g- L# c& L( b1 @9 @7 \2 d/ z- L
背景音乐,毫无疑问得要啦!格式为MOD。7 D' I. E9 e! V x3 ^
7 w7 L* ~% E4 h9 p: d程序开始。包含头文件,包含资源文件。
+ K a* T6 C7 Y7 h" S; d; C( |
; E# \, u* |3 q( n& R% @4 k主程序,打开中断,然后直接进入死循环。所有的工作,都留给中断去做。
, s! a* o/ Z g' i( X3 f' m4 v. N) ?+ N9 D
由于中断要处理各种不同的函数,所以需要在程序中变换中断服务程序。有两种方法可以实现:函数指针和分支。/ s7 B0 R e; O9 L; O! R
. W8 p- a6 Q& j7 X% J% N' r- S8 L函数指针方面,我没有学过,所以我也没办法讲解了。我就用分支来处理这种情况吧。顺便一说,用分支并不是一个好方法,如果游戏比较复杂,那么分支将会非常庞大;到时候恐怕连你自己都不明白到底是怎么回事了……
, A; ` \& e/ G- q% F8 e" i4 o A$ B! y5 O3 o& B7 W2 m J5 `1 l8 j
所谓分支,就是通过一个游戏流程变量(我一般称之为“剧情变量”)来控制游戏当前状况。这个变量为全局变量,初值为0。中断服务程序就是通过这个变量的值,来选择不同的中断服务。就拿咱们这次的游戏为例,游戏刚开始,显然要显示你的LOGO,因此在中断服务中,case 0的时候就是载入LOGO的数据,然后等待同步。中断完成之后顺便让剧情变量增值一下,那么下一次中断来到的时候,执行的就是case 1的服务。很明显这个服务程序就该是FADE特效了……8 p* Q3 G6 d2 @- m- B
# o" `% W0 `, ~. J( n( I. I! i& ^5 C代码中出现了一些有趣的技巧,如果你不明白,不妨把你不明白的地方屏蔽掉,重新编译看看出现了什么效果,或许你会知道我为什么会这样做。: R7 G. B0 _/ q; ], F* f3 H8 p4 _
: {2 F/ r, o6 ^' h5 W
游戏中出现了一个明显的问题,就是背景音乐。背景音乐在标题菜单中播放效果非常好,但是一旦游戏开始,数字开始变化的时候,音乐明显出现“卡”的现象。这是怎么回事呢?这与我们写程序的方式有关系。我们的主程序中,什么都没有,所有的工作,都是交给了中断去做。中断,顾名思义就是中断主程序,转向服务程序。但是音乐正在播放,这个时候中断主程序而转向服务程序,音乐自然也被中断了。不过,由于服务程序非常短小,所以中断服务被迅速执行完毕然后退出来继续执行主程序,也就是继续播放音乐,所以,从宏观上看,就是音乐出现了一个短暂的中断,从而出现了“卡”的现象。
5 T0 E, G) @% s5 P w* ?9 I$ V( I3 a2 Z7 {
那么怎么解决这个问题呢?没有别的办法,重写程序,让主程序不要吃闲饭,尽量少占用中断,非得占用中断的话,尽量缩短中断的长度。如果非得大量占用中断的话,那就只有抛弃MOD PLAYER,使用MP2K了!
+ l/ F/ Z* a$ _% q2 w! w C9 u6 t- T
程序没有优化,因此冗余代码很多。你要是有能力,优化一下会减小ROM体积。程序中还存在一些预留的入口,利用好了,可以做出“秘籍”、“隐藏要素”!当然,你也可以继续扩大程序,加上其它的功能。比如纪录通关的按键次数,然后用存档函数保存数据;新写一个界面,做成排行榜,然后利用排序函数做出排名……毛主席说过:人有多大胆,地有多大产!虽然这话有点唯心主义,但是可以确定的是,只要你有想法,只要你有能力,没有你实现不了的东西…… | * b$ W4 X+ f9 ~/ Z2 s
' h, J' \; Q$ A) q8 |3 J- |2 F- h7 {
9 P5 {, D9 u+ q
; A$ S. P, P. ^. H1 ?& q |
|