GamePage.dart 6.2 KB

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