跟我學(xué)Java Swing之游戲設(shè)計(jì)(2)
文章來源:電腦愛好者 作者:張劍
還記得《偷天換日》中精靈般穿梭在好萊塢車流中的Minicooper嗎?馬克·沃爾伯格和莎莉·賽隆就是駕駛著它在仇人的鼻子底下運(yùn)走了價(jià)值千萬的黃金??墒牵绻F(xiàn)在將一輛無法奔馳的Minicooper軀殼放在你的面前,你會(huì)如何看待它?它還是那個(gè)游走自如的精靈嗎?今天,就讓我們一點(diǎn)一點(diǎn)地為這輛Minicooper組裝上零件,讓它跑起來。
前言
從本期開始,我們?yōu)榇蠹姨峁┩暾挠螒蛟创a(點(diǎn)擊下載)。Java咖啡館倡導(dǎo)大家理論與實(shí)踐并重,我們?cè)谶B載中將給大家介紹關(guān)鍵技術(shù)以及實(shí)現(xiàn)思路,朋友們自行結(jié)合文章閱讀源代碼,好比一邊讀報(bào)一邊喝咖啡,這才是滴滴香濃意猶未盡。
游戲布局
“連連看”屬于一款二維戰(zhàn)棋類游戲,要設(shè)計(jì)棋盤類的游戲,GridLayout應(yīng)該是不二之選?,F(xiàn)在讓我們一起來看看GridLayout的構(gòu)造函數(shù):
·GridLayout():默認(rèn)的情況下,將布局區(qū)域劃分為1*1的大小
·GridLayout(int rows,int cols):指定布局區(qū)域橫向和縱向的格子數(shù)
·GridLayout(int rows,int cols,int hgap,int vgap):同上,并且還指定了每個(gè)格子之間的橫向間距hgap和縱向間距vgap
千萬別讓這三個(gè)構(gòu)造函數(shù)把你給嚇住了,其實(shí)只要你喜歡,完全可以放心大膽地使用其中的任何一個(gè),就算不小心用“錯(cuò)”了,以后也有辦法進(jìn)行調(diào)整。惟一需要注意的是,GridLayout在添加控件時(shí),默認(rèn)順序是從左上方向右下方依次添加的。
現(xiàn)在讓我們來確定游戲的格子數(shù)目。究竟多少格子比較合適呢?太少會(huì)降低游戲的難度,太多又會(huì)造成視覺影響。所以,我們應(yīng)該通過一對(duì)常量來表示,將來即使要修改,也是舉手之勞。
在Java中,常量的定義需要寫成public final static的形式,假如我們規(guī)定游戲的棋盤在橫向有8個(gè)格子,縱向也有8個(gè)格子,那么,我們應(yīng)該這樣定義:
public final static int ROW = 8;
public final static int COLUMN = 8;
然后,我們使用GridLayout的第二種構(gòu)造函數(shù)來創(chuàng)建布局:
GridLayout gridLayout = new GridLayout(ROW, COLUMN);
最后,我們還需要將游戲區(qū)(contentPanel)的布局改為上述布局:
contentPanel.setLayout(gridLayout);
如果你此時(shí)編譯并運(yùn)行程序的話,你可能會(huì)奇怪:界面怎么沒有發(fā)生任何改變,是不是哪出錯(cuò)了?雖然我們指定了布局,可是什么控件也沒有添加,當(dāng)然就看不出變化?,F(xiàn)在讓我們一起在布局上添加按鈕吧:
for (int i = 0; i < ROW * COLUMN; i++) {
JButton button = new JButton("Kyodai");
contentPanel.add(button);
}
再運(yùn)行程序試試,是不是和我的一樣(見圖1)?
巧用JButton做文章
JButton是一個(gè)按鈕控件,它也是Swing中普通得不能再普通的控件了,盡管如此,我們還是需要花費(fèi)一點(diǎn)功夫來了解和使用它,因?yàn)楫?dāng)你能夠熟練使用JButton后,你會(huì)發(fā)現(xiàn)其他的Swing控件也是如此的相似。
如果你將剛才寫好的程序拿來運(yùn)行,你會(huì)發(fā)現(xiàn):游戲區(qū)的按鈕總是排得滿滿的,這對(duì)實(shí)際游戲的操作非常不便,所以,我們得想辦法讓一部分格子空出來。GridLayout布局什么都好,就是在添加控件的時(shí)候不能跳過某一個(gè)格子,這下可怎么辦呢?
其實(shí)這也不難,既然GridLayout不讓跳過,如果我們讓某個(gè)格子內(nèi)添加的控件與GridLayout布局的背景融為一體,這樣在視覺上就達(dá)到了一致的效果。此外,假如別人在無意中點(diǎn)擊到這個(gè)格子上,按鈕仍然就會(huì)原形畢露,我們還得想辦法讓按鈕不能被點(diǎn)擊,這就需要用到JButton的setEnabled()方法。最后,對(duì)于能夠點(diǎn)擊的按鈕,當(dāng)它們被點(diǎn)擊時(shí),我們還得要區(qū)分出來究竟是哪一個(gè)按鈕被點(diǎn)擊了。
在上一次實(shí)現(xiàn)“關(guān)于”功能的時(shí)候,我們使用了e.getSource()方法來判斷鼠標(biāo)點(diǎn)擊事件產(chǎn)生的源,然而,那只對(duì)已經(jīng)命名好了的控件比較有效。這里,使用數(shù)組表示按鈕無疑是最好的方法了,首先讓我們將上面的代碼修改一下:
JButton[] dots = new JButton[ROW * COLUMN];
for (int i = 0; i < ROW * COLUMN; i++) {
dots[i] = new JButton("Kyodai");
dots[i].setActionCommand("" + i);
contentPanel.add(dots[i]);
dots[i].addActionListener(this);
}
千萬別忘記了在循環(huán)體中寫上dots[i] = new JButton("Kyodai"),雖然在前面定義、使用了dots組數(shù),然而,這僅僅只是告訴程序我們需要使用一些JButton,但是,這些JButton卻依然沒有被初始化。同時(shí),我們不僅使用setActionCommand()為按鈕制定了事件名稱,還使用了addActionListener()方法為每個(gè)按鈕加上了事件響應(yīng)處理。
關(guān)于事件響應(yīng)的代碼,我們可以在原來actionPerformed()事件代碼的后面添加:
if (e.getSource() instanceof JButton) {
JButton button = (JButton) e.getSource();
int offset = Integer.parseInt(button.getActionCommand());
int row, col;
row = Math.round(offset / COLUMN);
col = offset - row * COLUMN;
JOptionPane.showMessageDialog(this,"ROW="+row+",COL="
+ col, "按鈕", JOptionPane.INFORMATION_MESSAGE);
}
在上面的代碼中,e.getSource() instanceof JButton用來判斷產(chǎn)生的事件是否是由JButton型的控件產(chǎn)生的,然后又將產(chǎn)生事件源的控件進(jìn)行強(qiáng)制型類轉(zhuǎn)換,再使用Integer.parseInt(button.getActionCommand())方法將取得的事件名稱轉(zhuǎn)化為整數(shù),后面的代碼就將這個(gè)整數(shù)還原成行和列的信息。
好了,現(xiàn)在運(yùn)行程序,然后點(diǎn)擊每個(gè)按鈕,看看是否會(huì)出現(xiàn)如右圖的對(duì)話框?
注意哦,我們的下標(biāo)是從0開始的。本期程序源代碼(點(diǎn)擊下載)。
在Swing中使用圖片
目前我們已經(jīng)解決了用戶操作的問題。為了讓界面美觀起來,我們需要使用圖片來代替文字。有時(shí)間和耐心的朋友可以自己做個(gè)性圖片,只是要注意保持每張圖片大小一致,否則就太難看啦。想省事的話也可以直接使用下載包中提供的圖片。
在Swing中,可以使用setIcon(Image image)方法來為JButton設(shè)置圖片,其中的參數(shù)image就是我們要設(shè)置的圖片對(duì)象,這個(gè)對(duì)象有多種方法可以得到:
·使用Toolkit類來獲得:
image = Toolkit.getDefaultToolkit().getImage(url);
·使用ImageIcon類的getImage()方法來獲得:
image = new ImageIcon(getClass().getResource(filename)).getImage();
我們這里選用第一種方法。為了方便今后再次獲得Image對(duì)象,我們可以將此寫為一個(gè)函數(shù):
Image getImage(String filename){
URLClassLoader urlLoader = (URLClassLoader) this.getClass().getClassLoader();
URL url = null;
Image image = null;
url = urlLoader.findResource(filename);
image = Toolkit.getDefaultToolkit().getImage(url);
return image;
}
有了這個(gè)函數(shù)之后,以后再要用到圖片就方便多了,不是嗎?
圖片現(xiàn)在已經(jīng)有了,那么我們就可以將源代碼包中Map.Java中的地圖信息用圖片的形式表現(xiàn)出來了。由于此段代碼是關(guān)于游戲的算法代碼,因此不再列舉代碼,在這里大致解釋一下具體是怎么做的吧!在Map.Java中,我們使用了二維數(shù)組map[][]來保存圖片信息,同時(shí),我們也使用了一維數(shù)組images[]來保存每一個(gè)圖片對(duì)象,map中的每一個(gè)元素都有一個(gè)值,這個(gè)值不僅表明了在游戲界面中按鈕對(duì)應(yīng)的值,也表明了這個(gè)按鈕使用images[]數(shù)組中圖片的編號(hào),程序運(yùn)行界面便漂亮許多。
小結(jié)
今天我們介紹了Swing中JButton控件。對(duì)于大多數(shù)Swing控件來說,JButton的用法一樣可以照搬過去。別小看這個(gè)JButton,當(dāng)你能夠很好地掌握它的時(shí)候,你的Swing功底就已經(jīng)提升一個(gè)臺(tái)階了。此外,我們也學(xué)會(huì)了在Java中加載圖片文件的兩種方法。最后,我們還將以上的兩部分內(nèi)容結(jié)合在一起創(chuàng)造出了漂亮的游戲界面。雖然我沒有在這里寫出完整的代碼,但相信大家參考我的源程序,經(jīng)過自己的努力,一定能夠達(dá)到目標(biāo)。如果遇到解決不了的問題,可以通過leftmoon@163.com信箱與我聯(lián)系。
不知道大家對(duì)今天的內(nèi)容感覺到滿意嗎?精彩還在后面,大家要繼續(xù)光顧我們的咖啡館哦!
上一篇:EJB輕松進(jìn)階之一
欄 目:Java編程
下一篇:面向?qū)ο缶幊?Java中的抽象數(shù)據(jù)類型
本文標(biāo)題:跟我學(xué)Java Swing之游戲設(shè)計(jì)(2)
本文地址:http://mengdiqiu.com.cn/a1/Javabiancheng/8551.html
您可能感興趣的文章
- 01-10Java咖啡館(1)——嘆咖啡
- 01-10Java Socket編程(三) 服務(wù)器Sockets
- 01-10Java進(jìn)階:Struts多模塊的技巧
- 01-10Java Socket編程(一) Socket傳輸模式
- 01-10Java Socket編程(二) Java面向連接的類
- 01-10Java運(yùn)行時(shí)多態(tài)性的實(shí)現(xiàn)
- 01-10Java經(jīng)驗(yàn)點(diǎn)滴:處理沒有被捕獲的異常
- 01-10Java Socket編程(四) 重復(fù)和并發(fā)服務(wù)器
- 01-10Java中的浮點(diǎn)數(shù)分析
- 01-10面向?qū)ο缶幊?Java中的抽象數(shù)據(jù)類型


閱讀排行
本欄相關(guān)
- 01-10Java咖啡館(1)——嘆咖啡
- 01-10JVM的垃圾回收機(jī)制詳解和調(diào)優(yōu)
- 01-10Java Socket編程(三) 服務(wù)器Sockets
- 01-10Java進(jìn)階:Struts多模塊的技巧
- 01-10J2SE 1.5版本的新特性一覽
- 01-10Java Socket編程(一) Socket傳輸模式
- 01-10Java運(yùn)行時(shí)多態(tài)性的實(shí)現(xiàn)
- 01-10Java Socket編程(二) Java面向連接的類
- 01-10Java Socket編程(四) 重復(fù)和并發(fā)服務(wù)
- 01-10Java經(jīng)驗(yàn)點(diǎn)滴:處理沒有被捕獲的異常
隨機(jī)閱讀
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 01-10使用C語言求解撲克牌的順子及n個(gè)骰子
- 01-10C#中split用法實(shí)例總結(jié)
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 01-11ajax實(shí)現(xiàn)頁面的局部加載
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 04-02jquery與jsp,用jquery
- 01-10delphi制作wav文件的方法
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?