查看: 2238|回复: 0

[.NET开发] C# 实现连连看功能(推荐)

发表于 2018-2-18 20:04:14

本文是利用C#实现连连看的小例子,以供学习分享使用。

思路:

初始化布局(横竖十行十列,共100个单元格,每一个格一个按钮,背景图为水果图片,随机生成) 。

初始化对应棋盘(用二维数组表示【0表示空白,非0表示界面对象】)和页面相对应,同步操作。

判断点击的图片是否可以消掉(转化为二维数组【以水平方向,垂直方向,一个拐角,两个拐角的步骤进行判断】)。

如可以消掉,隐藏图片,增加分数。

时间限制,采用倒计时方式。

涉及知识点:

线程:Thread,后台运行时间控制【倒计时方式】。

界面闪烁:当界面中的控件较多,且有背景图时,界面就会出现闪烁【解决方式:1,双缓冲方式 2. 设置控件创建样式,统一刷新】。

TableLayoutPanel:表示一个面板,它可以在一个由行和列组成的网格中对其内容进行动态布局【新增元素,设置行列,以及样式】。

资源文件:Resources 用于存放图片及其他资源。

Button:FlatAppearance获取用于指示选中状态和鼠标状态的边框外观和颜色。

效果图图下(一)【开始,初始化后,倒计时功能,停止功能】:

效果图(二)【时间结束】

核心代码如下:

  1. /// <summary>
  2. /// 连连看帮助类
  3. /// </summary>
  4. public class LinkHelper
  5. {
  6. /// <summary>
  7. /// 连连看,看板
  8. /// </summary>
  9. public int[,] LinkBoard { get; set; }
  10. /// <summary>
  11. /// 连线成功事件
  12. /// </summary>
  13. public event EventHandler SucClick;
  14. /// <summary>
  15. /// 连接失败事件
  16. /// </summary>
  17. public event EventHandler FailClick;
  18. private int col = 10;
  19. public int Col
  20. {
  21. get
  22. {
  23. return col;
  24. }
  25. set
  26. {
  27. col = value;
  28. }
  29. }
  30. private int row = 10;
  31. public int Row
  32. {
  33. get
  34. {
  35. return row;
  36. }
  37. set
  38. {
  39. row = value;
  40. }
  41. }
  42. /// <summary>
  43. /// 尝试连线
  44. /// </summary>
  45. public void LinkLine(Point first, Point second)
  46. {
  47. EventArgs e = new EventArgs();
  48. if (checkLink(first, second))
  49. {
  50. //连线成功
  51. this.LinkBoard[first.X, first.Y] = 0;
  52. this.LinkBoard[second.X, second.Y] = 0;
  53. if (this.SucClick != null)
  54. {
  55. SucClick(this, e);
  56. }
  57. }
  58. else {
  59. //连线失败
  60. if (this.FailClick != null)
  61. {
  62. FailClick(this, e);
  63. }
  64. }
  65. }
  66. /// <summary>
  67. /// 是否赋值
  68. /// </summary>
  69. /// <param name="p"></param>
  70. /// <returns></returns>
  71. public bool IsChecked(Point p)
  72. {
  73. bool flag = false;
  74. if (p.X != -1 && p.Y != -1)
  75. {
  76. flag = true;
  77. }
  78. return flag;
  79. }
  80. #region 核心算法
  81. /// <summary>
  82. /// 判断是否连线成功
  83. /// </summary>
  84. /// <param name="a">第一个点击对象</param>
  85. /// <param name="b">第二个点击对象</param>
  86. /// <returns></returns>
  87. private bool checkLink(Point a, Point b)
  88. {
  89. if (!Point.Equals(a, b))
  90. {
  91. if (this.LinkBoard[a.X, a.Y] == this.LinkBoard[b.X, b.Y])
  92. {
  93. if (a.X == b.X && horizon(a, b))
  94. {
  95. return true;
  96. }
  97. if (a.Y == b.Y && vertical(a, b))
  98. {
  99. return true;
  100. }
  101. if (oneCorner(a, b))
  102. {
  103. return true;
  104. }
  105. else
  106. {
  107. return twoCorner(a, b);
  108. }
  109. }
  110. else {
  111. //如果点击的不是同一个图案,直接返回false
  112. return false;
  113. }
  114. }
  115. else {
  116. //如果点击的是同一个位置的图案,直接返回false;
  117. return false;
  118. }
  119. }
  120. /// <summary>
  121. /// 水平连线
  122. /// </summary>
  123. /// <param name="a"></param>
  124. /// <param name="b"></param>
  125. /// <returns></returns>
  126. private bool horizon(Point a, Point b)
  127. {
  128. int col_start = a.Y < b.Y ? a.Y : b.Y; //获取a,b中较小的y值
  129. int col_end = a.Y < b.Y ? b.Y : a.Y; //获取a,b中较大的值
  130. //遍历a,b之间是否通路,如果一个不是就返回false;
  131. for (int i = col_start + 1; i < col_end; i++)
  132. {
  133. if (this.LinkBoard[a.X, i] != 0)
  134. {
  135. return false;
  136. }
  137. }
  138. return true;
  139. }
  140. /// <summary>
  141. /// 垂直连线
  142. /// </summary>
  143. /// <param name="a"></param>
  144. /// <param name="b"></param>
  145. /// <returns></returns>
  146. private bool vertical(Point a, Point b)
  147. {
  148. int row_start = a.X < b.X ? a.X : b.X;
  149. int row_end = a.X < b.X ? b.X : a.X;
  150. for (int i = row_start + 1; i < row_end; i++)
  151. {
  152. if (this.LinkBoard[i, a.Y] != 0)
  153. {
  154. return false;
  155. }
  156. }
  157. return true;
  158. }
  159. /// <summary>
  160. /// 一个拐角
  161. /// </summary>
  162. /// <param name="a"></param>
  163. /// <param name="b"></param>
  164. /// <returns></returns>
  165. private bool oneCorner(Point a, Point b)
  166. {
  167. Point c = new Point(b.X, a.Y);
  168. Point d = new Point(a.X, b.Y);
  169. //判断C点是否有元素
  170. if (this.LinkBoard[c.X, c.Y] == 0)
  171. {
  172. bool path1 = horizon(b, c) && vertical(a, c);
  173. return path1;
  174. }
  175. //判断D点是否有元素
  176. if (this.LinkBoard[d.X, d.Y] == 0)
  177. {
  178. bool path2 = horizon(a, d) && vertical(b, d);
  179. return path2;
  180. }
  181. else
  182. {
  183. return false;
  184. }
  185. }
  186. /// <summary>
  187. /// 两个拐角
  188. /// </summary>
  189. /// <param name="a"></param>
  190. /// <param name="b"></param>
  191. /// <returns></returns>
  192. private bool twoCorner(Point a, Point b)
  193. {
  194. List<Line> ll = scan(a, b);
  195. if (ll.Count == 0)
  196. {
  197. return false;
  198. }
  199. for (int i = 0; i < ll.Count; i++)
  200. {
  201. Line tmpLine = ll[i];
  202. if (tmpLine.direct == 1)
  203. {
  204. if (vertical(a, tmpLine.a) && vertical(b, tmpLine.b))
  205. {
  206. return true;
  207. }
  208. }
  209. else if (tmpLine.direct == 0)
  210. {
  211. if (horizon(a, tmpLine.a) && horizon(b, tmpLine.b))
  212. {
  213. return true;
  214. }
  215. }
  216. }
  217. return false;
  218. }
  219. /// <summary>
  220. /// 扫描A与B之间的连接点组成的线
  221. /// </summary>
  222. /// <param name="a"></param>
  223. /// <param name="b"></param>
  224. /// <returns></returns>
  225. private List<Line> scan(Point a, Point b)
  226. {
  227. List<Line> linkList = new List<Line>();
  228. //检测a点,b点的左侧是否能够垂直直连
  229. for (int i = a.Y; i >= 0; i--)
  230. {
  231. if (this.LinkBoard[a.X, i] == 0 && this.LinkBoard[b.X, i] == 0 && vertical(new Point(a.X, i), new Point(b.X, i)))
  232. {
  233. linkList.Add(new Line(new Point(a.X, i), new Point(b.X, i), 0));
  234. }
  235. }
  236. //检测a点,b点的右侧是否能够垂直直连
  237. for (int i = a.Y; i < Col; i++)
  238. {
  239. if (this.LinkBoard[a.X, i] == 0 && this.LinkBoard[b.X, i] == 0 && vertical(new Point(a.X, i), new Point(b.X, i)))
  240. {
  241. linkList.Add(new Line(new Point(a.X, i), new Point(b.X, i), 0));
  242. }
  243. }
  244. //检测a点,b点的上侧是否能够水平直连
  245. for (int j = a.X; j >= 0; j--)
  246. {
  247. if (this.LinkBoard[j, a.Y] == 0 && this.LinkBoard[j, b.Y] == 0 && horizon(new Point(j, a.Y), new Point(j, b.Y)))
  248. {
  249. linkList.Add(new Line(new Point(j, a.Y), new Point(j, b.Y), 1));
  250. }
  251. }
  252. //检测a点,b点的下侧是否能够水平直连
  253. for (int j = a.X; j < Row; j++)
  254. {
  255. if (this.LinkBoard[j, a.Y] == 0 && this.LinkBoard[j, b.Y] == 0 && horizon(new Point(j, a.Y), new Point(j, b.Y)))
  256. {
  257. linkList.Add(new Line(new Point(j, a.Y), new Point(j, b.Y), 1));
  258. }
  259. }
  260. return linkList;
  261. }
  262. #endregion
  263. }
复制代码

以上所述是小编给大家介绍的C# 实现连连看功能,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对程序员之家网站的支持!

您可能感兴趣的文章:

  • 用javascript做一个webgame连连看大家看下
  • javascript 连连看代码出炉
  • 原生JavaScript实现连连看游戏(附源码)
  • JavaScript编写连连看小游戏


回复

使用道具 举报