您好,登錄后才能下訂單哦!
學C#的原因其實挺簡單的,因為一直對游戲挺感興趣,查了下比較流行的游戲引擎Unity的主要開發語言是C#,所以就決定從C#入手,學學面向對象的編程方法。
以前基本都做的是嵌入式開發,做嵌入式久了,基本上只用C語言,C語言面向過程的特性在嵌入式編程這種資源極度受限的情況確實十分有利,但這種方式在面對大型軟件的開發的時候就很難勝任了。編程的模式其實是一種思維習慣,習慣久了以后,想改變確實是一個艱難的過程···
說起C#,其實在大學的時候學過一個學期,說來慚愧那時候倒也沒把它當一門面向對象的語言(其實是當時根本不知道面向對象是啥),感覺跟C語言也就一點語法差異,把所有的用法全部歸為語法不同,說來也神奇,這種方法倒也能編程。最終學期結束的時候交上去一份用Winform開發的掃雷游戲結束了我的C#學習,在那之后就再也沒碰過C#。
現在重拾C#,為了免除掉不必要的干擾,并沒有直接在Unity上學習,而是仍然在VS中學習,但這次選擇了比較新的WPF,而不是WInform,作為學習,第一個任務還是跟以前一樣做一個掃雷游戲。
寫在不怎么前面的前面:本文主要分享下程序分析過程,具體的實現方法不是本文重點,對實現有問題的朋友可以自行評論區留言索要源碼或者提問^_^。
一、分析
1.游戲分析
那進入正題,應該如何完成這個游戲。忽略細枝末節的部分(如計時,顯示剩余雷數,菜單欄等)不說,就單說這個游戲的主體:掃雷區。
在游戲沒開始的時候,掃雷區放眼望去其實只有一個東西,那就是方塊...
忽略光影效果不談(是的,我又忽略了···),所有方塊的顏色都一樣,都響應相同的事件,那就是左鍵和右鍵。左鍵點開方塊,右鍵給方塊做個標記,認定為地雷。再繼續分析,方塊具有不同的種類。有的方塊點開之后周圍會有一大片方塊一起打開。有的方塊下面是地雷,點開就GameOver。還有方塊下面是數字,代表著周圍有多少個地雷。(果然,我又忽略了鼠標兩個鍵同時按自動打開周圍格子和第二次右鍵可以顯示問號的功能···但其實之后會發現這個功能其實要增加也會很簡單)。
所以,先來總結下掃雷游戲實現的核心:
2.實現技術分析
經過分析,是不是發現掃雷的的玩法其實很簡單,實現的技術也不難,全是靜態的沒有動畫的存在。
方塊的表現很像一個只能按一次的按鈕(事實上,在大學的時候我就是直接繼承的按鈕控件)。
但這一次為了能使用到更多C#相關的東西我使用了更加麻煩的自定義控件的方式。
方塊有三種表現形式,為特殊性,但很顯然也具有共性,所以在設計的時候,我把按鈕共性抽離出來,設計成了一個抽象的基類Cube。方塊有三種類型,但因為我懶,所以把其中的兩種(空白和數字)合并為了NumCube類,包含地雷的為BombCube類,這兩個類分別繼承了Cube。
Cube的實現:
Cube類中擁有以下字段:
ImageSource cubeNormalPic ImageSource cubeOnPic ImageSource cubeDownPic ImageSource cubeDisablePic ImageSource cubeFlagPic
這5個字段是用來設置Cube在各個狀態所顯示的圖片的(普通,鼠標進入,左鍵按下,失能,標記)
Bool isEnable Bool isFlag
這兩個字段就是標記Cube是否被使能和Flag
Image cubeImageHigh Image cubeImageLow
這2個是兩個image控件,作用是用來顯示圖片,之所以要2個圖片是因為旗子圖片被設計為一個疊加在Cube上的圖片。
下面再來重點講下下面2個東西:
displayCube mouseEvent
在設計中,這是兩個接口,分別用來處理鼠標事件和方塊的展開。不同于直接在內部直接實現接口,將兩個接口設計為Cube屬性是為了能動態的修改這兩個接口的實現方式,不至于每次修改都需要對Cube內的代碼進行修改,且可以實現每個不同的Cube都使用不同的代碼而不需要使用重寫,這種方式在設計模式中也叫“策略模式”。
Cube只擁有一個方法,那就是Open,但這個方法其實也是有display接口代理實現。
public void Open() { if (displayCube != null) { displayCube.Open(this); } }
displayCube.Open(this)之所以要把自身傳入,是因為Open方法要用到Cube自己的參數和方法。
BombCube繼承自Cube
只添加了一個字段:
ImageSource bombPic
用來存儲地雷圖片.
NumCube 繼承自Cube
Int bombNum
用來記錄方塊周圍有多少個BombCube,當其為0的時候,NumCube就是顯示為空的方塊。
添加了一個組件lable用來顯示數字Text。
interface的實現
分別為每種Cube設計了一種接口的實現方式,使用這種方式,若后期需要改為動畫顯示,也只需要實現一個動畫的接口,賦值給對應的Cube就可以了。
二、實現
控件繼承:
Wpf進行控件繼承的時候需要注意,被繼承的控件不能有xaml。
在繼承的時候,xaml中需要加入如下語句:
< myTypes:Cube x:Class="掃雷.UserControl.NumCube" xmlns=" http:// schemas.microsoft.com/w infx/2006/xaml/presentation " xmlns:x=" http:// schemas.microsoft.com/w infx/2006/xaml " xmlns:mc=" http:// schemas.openxmlformats.org /markup-compatibility/2006 " xmlns:d=" http:// schemas.microsoft.com/e xpression/blend/2008 " mc:Ignorable="d" xmlns:myTypes="clr-namespace:掃雷.UserControl" d:DesignHeight="18" d:DesignWidth="18">
Cube 鼠標事件的實現:
鼠標事件主要是在各個事件中實現對Cube圖片的變換,例如鼠標移出事件
public void MouseLeaveCube(object sender, MouseEventArgs e) { BombCube bombCube = sender as BombCube; if (bombCube.IsEnable) { isClicking = false; bombCube.cubeImageLow.Source = bombCube.cubeNormalPic; } }
關于地雷位置的生成算法實現:
游戲很重要的一個方面是,每次地雷的位置應該不同。很容易想到應該用隨機數來產生地雷的位置。這就需要隨機生成N個不相同的坐標。本程序的實現方法是創建一個list<int>,之后使用隨機數在0-sizeX * sizeY - 1之間隨機生成一個數,檢查list中是否包含該數字,若不包含則添加進list,直到list擁有N個元素停止。
List<int> BombIndexList=new List<int>(); Random ran = new Random(); do { int bombIndex = ran.Next(0,sizeX * sizeY - 1); if(!BombIndexList.Contains(bombIndex)) { BombIndexList.Add(bombIndex); } else { continue; } } while (BombIndexList.Count < BombNum); IndexList = BombIndexList;
之后根據生成的list來確定坐標上應該是NumCube還是BombCube
for (int y = 0; y < sizeY; y++) { for (int x = 0; x < sizeX;x++) { //cube屬性設置 if(bombIndexList.Exists((int temp) => temp == x + y * cubeX)) { cubexMatrix[x, y] =bombCubeList[bombIndex++]; } else { numCubeList[numIndex].Text =""; cubexMatrix[x, y] =numCubeList[numIndex++]; } cubexMatrix[x, y].IsFlag =false; cubexMatrix[x, y].Margin =new Thickness(x * 18, y * 18, 0, 0); cubexMatrix[x, y].IsEnable = true; SetCubeBombNum(cubexMatrix,cubeX, cubeY); bombGrid.Children.Add(cubexMatrix[x, y]); } }
如何讓空白Cube打開以后會打開周圍的Cube:
因為這種打開方式有點類似于遞歸,需要有傳染性(即若打開的也是空白Cube,則其也應該打開周圍的Cube),所以執行該事件的時候一定要具有周圍Cube的信息(即能獲取到周圍的控件)。
獲取周圍的Cube的方法有兩種:
1.保存Cube自身的位置,并獲取所有Cube的位置
2.保存周圍Cube的信息
我使用的是第二種方式,之前Cube類中的Cubelist就是用來保存周圍Cube的信息的。通過CubeList找到周圍Cube,并觸發他們的左鍵單擊事件。
public void MouseLeftButtonUp(object sender, MouseButtonEventArgs e) { NumCube numCube = sender as NumCube; if (numCube.IsEnable && numCube.IsFlag == false) { // 完成在控件上點擊 if (isClicking) { isClicking = false; numCube.IsEnable = false; if (numCube.BombNum != 0) numCube.Text = Convert.ToString(numCube.BombNum); else { foreach (Cube cubeTemp in numCube.CubeList) { MouseButtonEventArgs args = new MouseButtonEventArgs(Mouse.PrimaryDevice, 0, MouseButton.Left); args.RoutedEvent = Cube.MouseLeftButtonDownEvent; cubeTemp.RaiseEvent(args); args.RoutedEvent = Cube.MouseLeftButtonUpEvent; cubeTemp.RaiseEvent(args); } } } } }
一些小技巧:
1.可以把一些圖片的修改放在屬性的set內,例如disable的圖片。
public bool IsEnable { get { return isEnable; } set { isEnable = value; if (isEnable) { if (cubeNormalPic != null) cubeImageLow.Source = cubeNormalPic; } else { if (cubeDisablePic != null) cubeImageLow.Source = cubeDisablePic; } } }
2.Wpf創建控件較慢,為了提升(修改寬度長度或地雷數量之后)游戲開始速度,應該預先創建控件,并把控件放入list或者arr保存,按照需求取出。
到這掃雷游戲的制作就沒什么難度技術上的難度的,只需要通過百度了解一些WPF常用的事件,控件,xalm相關的知識就能做出一個掃雷游戲啦。相關源碼就不發在這了,需要的朋友可以評論中找我,這次游戲制作讓我對面向對象的基本編程方法的了解有了一個很大的提升,下次應該就可以在Unity中做游戲啦 哈哈。
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持億速云。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。