GamePage.dart 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237
  1. import 'dart:math';
  2. import 'package:flutter/cupertino.dart';
  3. import 'package:flutter/material.dart';
  4. import 'package:gobang/ai/Ai.dart';
  5. import 'package:gobang/flyweight/Chess.dart';
  6. import 'package:gobang/flyweight/ChessFlyweightFactory.dart';
  7. import 'package:gobang/utils/TipsDialog.dart';
  8. import 'flyweight/Position.dart';
  9. ///简单的实现五子棋效果
  10. class GamePage extends StatefulWidget {
  11. @override
  12. State<StatefulWidget> createState() => GamePageState();
  13. }
  14. class GamePageState extends State<GamePage> {
  15. @override
  16. void initState() {
  17. super.initState();
  18. }
  19. @override
  20. Widget build(BuildContext context) {
  21. return Scaffold(
  22. appBar: AppBar(
  23. title: Text("南瓜五子棋"),
  24. ),
  25. body: Center(
  26. child: Column(
  27. mainAxisAlignment: MainAxisAlignment.center,
  28. crossAxisAlignment: CrossAxisAlignment.center,
  29. mainAxisSize: MainAxisSize.max,
  30. children: <Widget>[
  31. Padding(
  32. padding: EdgeInsets.only(bottom: 15.0),
  33. child: CupertinoButton.filled(
  34. padding: EdgeInsets.all(0.0),
  35. child: Text("重置棋盘"),
  36. onPressed: () {
  37. setState(() {
  38. ChessPainter._position = null;
  39. ChessPainter._positions = [];
  40. Ai.getInstance().init();
  41. // blackChess = null;
  42. });
  43. }),
  44. ),
  45. Padding(
  46. padding: EdgeInsets.only(bottom: 15.0),
  47. child: CupertinoButton.filled(
  48. padding: EdgeInsets.all(0.0),
  49. child: Text("Ai下棋"),
  50. onPressed: () {
  51. turnAi();
  52. // blackChess = null;
  53. }),
  54. ),
  55. GestureDetector(
  56. onTapDown: (topDownDetails) {
  57. var position = topDownDetails.localPosition;
  58. Chess chess;
  59. if (ChessPainter._state == 0) {
  60. chess =
  61. ChessFlyweightFactory.getInstance().getChess("white");
  62. } else {
  63. chess =
  64. ChessFlyweightFactory.getInstance().getChess("black");
  65. }
  66. setState(() {
  67. ChessPainter._position = Position(position.dx, position.dy, chess);
  68. });
  69. },
  70. child: Stack(
  71. children: [
  72. CustomPaint(
  73. size: Size(300.0, 300.0),
  74. painter: CheckerBoardPainter(),
  75. ),
  76. CustomPaint(
  77. size: Size(300.0, 300.0),
  78. painter: ChessPainter(turnAi),
  79. )
  80. ],
  81. ))
  82. ]),
  83. ),
  84. );
  85. }
  86. void turnAi() {
  87. if(ChessPainter._position!.chess is WhiteChess && Ai.getInstance().isWin(ChessPainter._position!.dx~/(300/15), ChessPainter._position!.dy~/(300/15), 1)){
  88. TipsDialog.show(context, "恭喜", "您打败了决策树算法");
  89. }
  90. Ai ai = Ai.getInstance();
  91. print("Owner:"+Ai.getInstance().isWin(ChessPainter._position!.dx~/(300/15), ChessPainter._position!.dy~/(300/15), 1).toString());
  92. ChessPainter._position = ai.searchPosition();
  93. Ai.getInstance().addChessman(ChessPainter._position!.dx.toInt(), ChessPainter._position!.dy.toInt(), -1);
  94. print("Ai:"+Ai.getInstance().isWin(ChessPainter._position!.dx.toInt(), ChessPainter._position!.dy.toInt(), -1).toString());
  95. if(ChessPainter._position!.chess is BlackChess &&Ai.getInstance().isWin(ChessPainter._position!.dx.toInt(), ChessPainter._position!.dy.toInt(), -1)){
  96. TipsDialog.show(context, "很遗憾", "决策树算法打败了您");
  97. }
  98. setState(() {
  99. ChessPainter._position!.dx = ChessPainter._position!.dx*(300/15);
  100. ChessPainter._position!.dy = ChessPainter._position!.dy*(300/15);
  101. });
  102. }
  103. }
  104. class ChessPainter extends CustomPainter {
  105. static List<Position> _positions = [];
  106. static int _state = 0;
  107. static Position? _position;
  108. final Function _function;
  109. ChessPainter(Function f):_function = f;
  110. @override
  111. void paint(Canvas canvas, Size size) {
  112. if (_position == null) {
  113. return;
  114. }
  115. bool add = false;
  116. double mWidth = size.width / 15;
  117. double mHeight = size.height / 15;
  118. var mPaint = Paint();
  119. //求两个点之间的距离,让棋子正确的显示在坐标轴上面
  120. var dx = _position!.dx;
  121. var dy = _position!.dy;
  122. for (int i = 0; i < CheckerBoardPainter._crossOverBeanList.length; i++) {
  123. var absX = (dx - CheckerBoardPainter._crossOverBeanList[i]._dx).abs(); //两个点的x轴距离
  124. var absY = (dy - CheckerBoardPainter._crossOverBeanList[i]._dy).abs(); //两个点的y轴距离
  125. var s = sqrt(absX * absX +
  126. absY * absY); //利用直角三角形求斜边公式(a的平方 + b的平方 = c的平方)来计算出两点间的距离
  127. if (s <= mWidth / 2 - 2) {
  128. // 触摸点到棋盘坐标坐标点距离小于等于棋子半径,那么
  129. //找到离触摸点最近的棋盘坐标点并记录保存下来
  130. _position!.dx = CheckerBoardPainter._crossOverBeanList[i]._dx;
  131. _position!.dy = CheckerBoardPainter._crossOverBeanList[i]._dy;
  132. _positions.add(_position!);
  133. add = true;
  134. if (_position!.chess is WhiteChess) {
  135. Ai.getInstance().addChessman(_position!.dx~/(300/15), _position!.dy~/(300/15), 1);
  136. }
  137. // flag = false; //白子下完了,该黑子下了
  138. break;
  139. }
  140. }
  141. //画子
  142. mPaint..style = PaintingStyle.fill;
  143. if (_positions.isNotEmpty) {
  144. for (int i = 0; i < _positions.length; i++) {
  145. mPaint..color = _positions[i].chess.color;
  146. canvas.drawCircle(Offset(_positions[i].dx, _positions[i].dy),
  147. min(mWidth / 2, mHeight / 2) - 2, mPaint);
  148. }
  149. }
  150. WidgetsBinding.instance!.addPostFrameCallback((_) {
  151. if (add &&_position!.chess is WhiteChess) {
  152. _function();
  153. }
  154. });
  155. }
  156. //在实际场景中正确利用此回调可以避免重绘开销,本示例我们简单的返回true
  157. @override
  158. bool shouldRepaint(CustomPainter oldDelegate) {
  159. // TODO: implement shouldRepaint
  160. return true;
  161. }
  162. }
  163. class CheckerBoardPainter extends CustomPainter {
  164. static List<CrossOverBean> _crossOverBeanList = [];
  165. static int _state = 0;
  166. @override
  167. void paint(Canvas canvas, Size size) {
  168. double mWidth = size.width / 15;
  169. double mHeight = size.height / 15;
  170. var mPaint = Paint();
  171. _crossOverBeanList.clear();
  172. canvas.drawColor(
  173. CupertinoColors.systemGrey, BlendMode.colorBurn); //重绘下整个界面的画布北京颜色
  174. //设置画笔,画棋盘背景
  175. mPaint
  176. ..isAntiAlias = true //抗锯齿
  177. ..style = PaintingStyle.fill //填充
  178. ..color = Color(0x77cdb175); //背景为纸黄色
  179. canvas.drawRect(
  180. Rect.fromCenter(
  181. center: Offset(size.width / 2, size.height / 2),
  182. width: size.width,
  183. height: size.height),
  184. mPaint);
  185. //画棋盘网格
  186. mPaint
  187. ..style = PaintingStyle.stroke
  188. ..color = CupertinoColors.systemGrey6
  189. ..strokeWidth = 1.0;
  190. for (var i = 0; i <= 15; i++) {
  191. //画横线
  192. canvas.drawLine(
  193. Offset(0, mHeight * i), Offset(size.width, mHeight * i), mPaint);
  194. }
  195. for (var i = 0; i <= 15; i++) {
  196. //画竖线
  197. canvas.drawLine(
  198. Offset(mWidth * i, 0), Offset(mWidth * i, size.height), mPaint);
  199. }
  200. //记录横竖线所有的交叉点
  201. for (int i = 0; i <= 15; i++) {
  202. for (int j = 0; j <= 15; j++) {
  203. _crossOverBeanList.add(CrossOverBean(mWidth * j, mHeight * i));
  204. }
  205. }
  206. }
  207. //在实际场景中正确利用此回调可以避免重绘开销,本示例我们简单的返回true
  208. @override
  209. bool shouldRepaint(CustomPainter oldDelegate) {
  210. // TODO: implement shouldRepaint
  211. return false;
  212. }
  213. }
  214. ///记录棋盘上横竖线的交叉点
  215. class CrossOverBean {
  216. double _dx;
  217. double _dy;
  218. CrossOverBean(this._dx, this._dy);
  219. }