练手WPF(三)——扫雷小游戏的简易实现(中)
八、随机布雷
/// <summary>
/// 随机布地雷
/// </summary>
/// <param name="mineNum">地雷数</param>
private void SetRndMine(int mineNum)
{
for (int k = ; k < mineNum; k++)
{
int nullnum = ; for (int j = ; j < _gameLevel._colGrid; j++)
{
for (int i = ; i < _gameLevel._rowGrid; i++)
{
if (_backData[j, i] == (int)BackState.BLANK)
nullnum++;
}
} if (nullnum < )
return; int index = rnd.Next(, nullnum);
nullnum = ;
for (int j = ; j < _gameLevel._colGrid; j++)
{
for (int i = ; i < _gameLevel._rowGrid; i++)
{
if (_backData[j, i] == )
{
nullnum++;
if (nullnum != index)
continue; _backData[j, i] = (int)BackState.MINE; // 设置为地雷
}
}
}
}
}
这个方法是不是很熟悉,我们在2048游戏中用到过,这里就不再多说了。 九、设置地雷旁边的格子标注的地雷数值
原理就是读取地雷位置,然后分别给不是地雷的格子数值+1。
/// <summary>
/// 设置地雷周围格子雷数
/// </summary>
private void SetCellMineNumber()
{
for (int y = ; y < _gameLevel._colGrid; y++)
{
for (int x = ; x < _gameLevel._rowGrid; x++)
{
if (_backData[y, x] == (int)BackState.MINE) // 遇到地雷则周围8格分别+1
{
if (x - > - && y - > - && _backData[y - , x - ] != (int)BackState.MINE)
_backData[y - , x - ]++; if (y - > - && _backData[y - , x] != (int)BackState.MINE)
_backData[y - , x]++; if (y - > - && x + < _gameLevel._rowGrid && _backData[y - , x + ] != (int)BackState.MINE)
_backData[y - , x + ]++; if (x - > - && _backData[y, x - ] != (int)BackState.MINE)
_backData[y, x - ]++; if (x + < _gameLevel._rowGrid && _backData[y, x + ] != (int)BackState.MINE)
_backData[y, x + ]++; if (y + < _gameLevel._colGrid && x - > - && _backData[y + , x - ] != (int)BackState.MINE)
_backData[y + , x - ]++; if (y + < _gameLevel._colGrid && _backData[y + , x] != (int)BackState.MINE)
_backData[y + , x]++; if (y + < _gameLevel._colGrid && x + < _gameLevel._rowGrid && _backData[y + , x + ] != (int)BackState.MINE)
_backData[y + , x + ]++;
}
}
}
}
十、设置背景区图片组
循环读取背景区数据,根据数值设置对应的图片源。
private void SetBackCellImage()
{
BackCanvas.Children.Clear(); for (int y=; y<_gameLevel._colGrid; y++)
{
for (int x=; x<_gameLevel._rowGrid; x++)
{
_backImage[y, x] = new Image();
if (_backData[y, x] == (int)BackState.BLANK)
{
_backImage[y, x].Source = ImageHelper.CutImage(_bmpSpace, new Int32Rect(, , _cellSize.Width, _cellSize.Height));
}
else if (_backData[y, x] == (int)BackState.MINE)
{
_backImage[y, x].Source = ImageHelper.CutImage(_bmpMine, new Int32Rect(, , _cellSize.Width, _cellSize.Height));
}
else
{
for (int i = ; i < ; i++)
{
if (_backData[y, x] == (i+))
{
_backImage[y, x].Source = ImageHelper.CutImage(
_bmpNum1_8, new Int32Rect(i * _cellSize.Width, , _cellSize.Width, _cellSize.Height));
break;
}
}
} _backImage[y, x].SetValue(Canvas.LeftProperty, x * (double)_cellSize.Width);
_backImage[y, x].SetValue(Canvas.TopProperty, y * (double)_cellSize.Height);
BackCanvas.Children.Add(_backImage[y, x]);
}
}
}
十一、开始游戏准备
根据游戏级别分别调用随机布雷、设置雷边数值方法,同时添加背景图片数组到游戏背景区。
private void ResetGame()
{
InitialGameState();
InitGameData(_level);
textBlockMineNum.Text = _gameLevel._mineNum.ToString(); SetRndMine(_gameLevel._mineNum);
SetCellMineNumber();
SetBackCellImage();
}
将该方法添加到开始游戏菜单中,并计时器打开。
private void MenuGameStart_Click(object sender, RoutedEventArgs e)
{
ResetGame();
_gameState = GameState.START; // 开始计时
_stopWatchGame.Start();
_timerSetTimeText.Start();
}
重复点击开始菜单,看看效果。
十二、调整游戏级别
根据菜单设置的当前级别值设置菜单选择状态。
/// <summary>
/// 设置级别菜单状态
/// </summary>
/// <param name="level"></param>
private void SetMenuLevelState(Level level)
{
switch (level)
{
case Level.SIMPLE:
MenuLowLevel.IsChecked = true;
MenuMiddleLevel.IsChecked = false;
MenuHighLevel.IsChecked = false;
break;
case Level.NORMAL:
MenuLowLevel.IsChecked = false;
MenuMiddleLevel.IsChecked = true;
MenuHighLevel.IsChecked = false;
break;
case Level.HARD:
MenuLowLevel.IsChecked = false;
MenuMiddleLevel.IsChecked = false;
MenuHighLevel.IsChecked = true;
break;
}
}
/// <summary>
/// 初级级别菜单
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void MenuLowLevel_Click(object sender, RoutedEventArgs e)
{
if (_level == Level.SIMPLE)
return; if (MessageBox.Show("将用新难度级别重置游戏,确认要继续吗?", "警告", MessageBoxButton.YesNo, MessageBoxImage.Asterisk)
== MessageBoxResult.Yes)
{
_level = Level.SIMPLE;
SetMenuLevelState(_level); ResetGame();
}
} /// <summary>
/// 中级级别菜单
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void MenuMiddleLevel_Click(object sender, RoutedEventArgs e)
{
if (_level == Level.NORMAL)
return; if (MessageBox.Show("将用新难度级别重置游戏,确认要继续吗?", "警告", MessageBoxButton.YesNo, MessageBoxImage.Asterisk)
== MessageBoxResult.Yes)
{
_level = Level.NORMAL;
SetMenuLevelState(_level);
ResetGame();
}
} /// <summary>
/// 高级级别菜单
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void MenuHighLevel_Click(object sender, RoutedEventArgs e)
{
if (_level == Level.HARD)
return; if (MessageBox.Show("将用新难度级别重置游戏,确认要继续吗?", "警告", MessageBoxButton.YesNo, MessageBoxImage.Asterisk)
== MessageBoxResult.Yes)
{
_level = Level.HARD;
SetMenuLevelState(_level);
ResetGame();
}
}
十三、设置前景区图片组
/// <summary>
/// 设置前景图片数组
/// </summary>
private void SetForeCellImages()
{
ForeCanvas.Children.Clear(); for (int y = ; y < _gameLevel._colGrid; y++)
{
for (int x = ; x < _gameLevel._rowGrid; x++)
{
if (_foreData[y, x] > (int)ForeState.NONE)
{
_foreImage[y, x] = new Image();
_foreImage[y, x].Source = ImageHelper.CutImage(_bmpForeground, new Int32Rect(, , _cellSize.Width, _cellSize.Height)); _foreImage[y, x].SetValue(Canvas.LeftProperty, x * (double)_cellSize.Width);
_foreImage[y, x].SetValue(Canvas.TopProperty, y * (double)_cellSize.Height);
ForeCanvas.Children.Add(_foreImage[y, x]);
}
}
}
}
在ResetGame()方法最后位置加入对该方法的调用。
运行程序,看看效果。
练手WPF(三)——扫雷小游戏的简易实现(中)的更多相关文章
- WEBGL学习笔记(七):实践练手1-飞行类小游戏之游戏控制
接上一节,游戏控制首先要解决的就是碰撞检测了 这里用到了学习笔记(三)射线检测的内容了 以鸟为射线原点,向前.上.下分别发射3个射线,射线的长度较短大概为10~30. 根据上一节场景的建设,我把y轴设 ...
- 练手WPF(三)——扫雷小游戏的简易实现(上)
一.创建项目1.创建WPF项目,设置初始化窗口大小(初级难度):高x宽为430x350.2.添加文件夹Images,并添加相关图片. 3.xaml中引入图片资源. <Window.Resourc ...
- 练手WPF(三)——扫雷小游戏的简易实现(下)
十四.响应鼠标点击事件 (1)设置对应坐标位置为相应的前景状态 /// <summary> /// 设置单元格图样 /// </summary> /// <para ...
- Angular4 扫雷小游戏
扫雷小游戏,可以升级过关,难度随关卡增加.但是有很明显的bug,以后有时间会继续优化! HTML: <div class="mainContent"> <div ...
- 扫雷小游戏PyQt5开发【附源代码】
也没啥可介绍哒,扫雷大家都玩过. 雷的分布算法也很简单,就是在雷地图(map:二维数组)中,随机放雷,然后这个雷的8个方位(上下左右.四个对角)的数字(非雷的标记.加一后不为雷的标记)都加一. 如何判 ...
- 练手WPF(二)——2048游戏的简易实现(上)
1.创建游戏界面编辑MainWindow.xaml,修改代码如下: <Window.Resources> <Style TargetType="Label"> ...
- web版扫雷小游戏(一)
作为一名程序猿,平时的爱好也不多,说起游戏,我不太喜欢大型的网游,因为太耗时间,偶尔玩玩经典的单机小游戏,比如windows下自带的游戏扫雷(秀一下,高级下最高纪录110s). 现阶段正在致力于web ...
- 练手WPF(一)——模拟时钟与数字时钟的制作(上)
一.Visual Studio创建一个WPF项目. 简单调整一下MainWindow.xaml文件.主要使用了两个Canvas控件,分别用于显示模拟和数字时钟,命名为AnalogCanvas.digi ...
- C++扫雷小游戏(基于CMD命令行)
这个小游戏是笔者在大一C语言课程设计的时候写的,基于命令行,为了显得漂亮一些,特别加上了彩色特效~~~ 注意:Win10系统须将命令行调为旧版命令行,否则有可能会显示乱码! 代码示例: #includ ...
随机推荐
- C#测试对比不同类型的方法调用的性能
一. 测试方法调用形式 1. 实例方法调用 2. 静态方法调用 3. 实例方法反射调用 4. 委托方法的Invoke调用 5. 委托方法的DynamicInvoke调用 6.委托方法的BeginInv ...
- Python读写Excel文件的实例
最近由于经常要用到Excel,需要根据Excel表格中的内容对一些apk进行处理,手动处理很麻烦,于是决定写脚本来处理.首先贴出网上找来的读写Excel的脚本. 1.读取Excel(需要安装xlrd) ...
- JS引擎和作用域、编译器之间对话
关于以下代码段: function foo(a) { console.log( a ); // 2 } foo( 2 ); JS引擎和作用域.编译器之间对话:
- 算法基础:BFS和DFS的直观解释
算法基础:BFS和DFS的直观解释 https://cuijiahua.com/blog/2018/01/alogrithm_10.html 一.前言 我们首次接触 BFS 和 DFS 时,应该是在数 ...
- 【XML】XML基本结构以及XML-Schema约束
XML 简介 1998年2月,W3C正式批准了可扩展标记语言的标准定义,可扩展标记语言可以对文档和数据进行结构化处理,从而能够在部门.客户和供应商之间进行交换,实现动态内容生成,企业集成和应用开发.可 ...
- xwiki 知识管理系统
搭建一个知识管理平台, 用于知识库管理/规范管理, 可以作wiki, 可以将word/excel等导入进去, 支持全文搜索, 可以记周报, 会议纪要. 现在有很多文档管理系统, 比如阿里的语雀.腾讯的 ...
- ElasticSearch安装及运行的坑
一.确认centos系统是为64位的,x86的不可以安装 1. 下载elasticsearch包 2. 用 tar -zxvf 解压包 3. 增加一个elk用户,elasticsearch7不可用ro ...
- Service__cmd安装MySQL并连接SQLyog
整理记录关于使用cmd安装mysql的过程 1.配置环境变量 1) 计算机->属性->高级系统设置->环境变量 2)先添加变量 变量名:MYSQL_HOME 变量值:D:\mys ...
- MySQL执行SQL脚本问题 :错误代码2006、1153
今天用mysql执行了一个60M的SQL脚本遇到了一些错误,经由网上查询如下: 1.#2006 - MySQL server has gone away 出现该错误代码原因如下: 1.应用程序长时间的 ...
- HDU 4729 An Easy Problem for Elfness(树链剖分边权+二分)
题意 链接:https://cn.vjudge.net/problem/HDU-4729 给你n个点,然你求两个点s和t之间的最大流.而且你有一定的钱k,可以进行两种操作 1.在任意连个点之间建立一个 ...