GamePage.dart 8.7 KB

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