Browse Source

完成所以模式设计,需要测试备忘录模式

JalorOMEN 3 years ago
parent
commit
aae99ab1c8

+ 66 - 35
lib/GamePage.dart

@@ -3,14 +3,19 @@ import 'dart:math';
 import 'package:flutter/cupertino.dart';
 import 'package:flutter/cupertino.dart';
 import 'package:flutter/material.dart';
 import 'package:flutter/material.dart';
 import 'package:gobang/ai/Ai.dart';
 import 'package:gobang/ai/Ai.dart';
+import 'package:gobang/bridge/ChessShape.dart';
 import 'package:gobang/factory/ThemeFactory.dart';
 import 'package:gobang/factory/ThemeFactory.dart';
 import 'package:gobang/flyweight/Chess.dart';
 import 'package:gobang/flyweight/Chess.dart';
 import 'package:gobang/flyweight/ChessFlyweightFactory.dart';
 import 'package:gobang/flyweight/ChessFlyweightFactory.dart';
+import 'package:gobang/memorandum/Originator.dart';
+import 'package:gobang/state/UserContext.dart';
 import 'package:gobang/utils/TipsDialog.dart';
 import 'package:gobang/utils/TipsDialog.dart';
+import 'package:gobang/viewModel/GameViewModel.dart';
 
 
 import 'flyweight/Position.dart';
 import 'flyweight/Position.dart';
 
 
 var width = 0.0;
 var width = 0.0;
+
 ///简单的实现五子棋效果
 ///简单的实现五子棋效果
 class GamePage extends StatefulWidget {
 class GamePage extends StatefulWidget {
   @override
   @override
@@ -18,8 +23,9 @@ class GamePage extends StatefulWidget {
 }
 }
 
 
 class GamePageState extends State<GamePage> {
 class GamePageState extends State<GamePage> {
-
   ThemeFactory? _themeFactory;
   ThemeFactory? _themeFactory;
+  GameViewModel _viewModel = GameViewModel.getInstance();
+  Originator _originator = Originator.getInstance();
 
 
   @override
   @override
   void initState() {
   void initState() {
@@ -29,11 +35,23 @@ class GamePageState extends State<GamePage> {
 
 
   @override
   @override
   Widget build(BuildContext context) {
   Widget build(BuildContext context) {
-    width = MediaQuery.of(context).size.width*0.8;
+    width = MediaQuery.of(context).size.width * 0.8;
     return Scaffold(
     return Scaffold(
       appBar: AppBar(
       appBar: AppBar(
         backgroundColor: _themeFactory!.getTheme().getThemeColor(),
         backgroundColor: _themeFactory!.getTheme().getThemeColor(),
         title: Text("南瓜五子棋"),
         title: Text("南瓜五子棋"),
+        actions: [
+          IconButton(
+              onPressed: () {
+                if(_viewModel.undo()){
+                  _originator.undo();
+                  setState(() {});
+                }else{
+                  TipsDialog.show(context, "提示", "现阶段不能悔棋");
+                }
+              },
+              icon: Icon(Icons.undo))
+        ],
       ),
       ),
       body: ListView(
       body: ListView(
         children: [
         children: [
@@ -51,7 +69,7 @@ class GamePageState extends State<GamePage> {
                         onPressed: () {
                         onPressed: () {
                           setState(() {
                           setState(() {
                             ChessPainter._position = null;
                             ChessPainter._position = null;
-                            ChessPainter._positions = [];
+                            _originator.clean();
                             Ai.getInstance().init();
                             Ai.getInstance().init();
                             // blackChess = null;
                             // blackChess = null;
                           });
                           });
@@ -63,23 +81,17 @@ class GamePageState extends State<GamePage> {
                         padding: EdgeInsets.all(0.0),
                         padding: EdgeInsets.all(0.0),
                         child: Text("Ai下棋"),
                         child: Text("Ai下棋"),
                         onPressed: () {
                         onPressed: () {
-                            turnAi();
-                            // blackChess = null;
+                          turnAi();
+                          // blackChess = null;
                         }),
                         }),
                   ),
                   ),
                   GestureDetector(
                   GestureDetector(
                       onTapDown: (topDownDetails) {
                       onTapDown: (topDownDetails) {
                         var position = topDownDetails.localPosition;
                         var position = topDownDetails.localPosition;
-                        Chess chess;
-                        if (ChessPainter._state == 0) {
-                          chess =
-                              ChessFlyweightFactory.getInstance().getChess("white");
-                        } else {
-                          chess =
-                              ChessFlyweightFactory.getInstance().getChess("black");
-                        }
+                        Chess chess = _viewModel.play();
                         setState(() {
                         setState(() {
-                          ChessPainter._position = Position(position.dx, position.dy, chess);
+                          ChessPainter._position =
+                              Position(position.dx, position.dy, chess);
                         });
                         });
                       },
                       },
                       child: Stack(
                       child: Stack(
@@ -101,32 +113,40 @@ class GamePageState extends State<GamePage> {
     );
     );
   }
   }
 
 
+  /// Ai 下棋
   void turnAi() {
   void turnAi() {
-    if(ChessPainter._position!.chess is WhiteChess && Ai.getInstance().isWin(ChessPainter._position!.dx~/(width/15), ChessPainter._position!.dy~/(width/15), 1)){
+    if (ChessPainter._position!.chess is WhiteChess &&
+        Ai.getInstance().isWin(ChessPainter._position!.dx ~/ (width / 15),
+            ChessPainter._position!.dy ~/ (width / 15), 1)) {
       TipsDialog.show(context, "恭喜", "您打败了决策树算法");
       TipsDialog.show(context, "恭喜", "您打败了决策树算法");
     }
     }
+    // 获取Ai下棋地址
     Ai ai = Ai.getInstance();
     Ai ai = Ai.getInstance();
-    print("Owner:"+Ai.getInstance().isWin(ChessPainter._position!.dx~/(width/15), ChessPainter._position!.dy~/(width/15), 1).toString());
     ChessPainter._position = ai.searchPosition();
     ChessPainter._position = ai.searchPosition();
-    Ai.getInstance().addChessman(ChessPainter._position!.dx.toInt(), ChessPainter._position!.dy.toInt(), -1);
-    print("Ai:"+Ai.getInstance().isWin(ChessPainter._position!.dx.toInt(), ChessPainter._position!.dy.toInt(), -1).toString());
-    if(ChessPainter._position!.chess is BlackChess &&Ai.getInstance().isWin(ChessPainter._position!.dx.toInt(), ChessPainter._position!.dy.toInt(), -1)){
+    // 设置棋子外观
+    ChessPainter._position!.chess.chessShape = CircleShape();
+    // 加入决策中
+    Ai.getInstance().addChessman(ChessPainter._position!.dx.toInt(),
+        ChessPainter._position!.dy.toInt(), -1);
+    if (ChessPainter._position!.chess is BlackChess &&
+        Ai.getInstance().isWin(ChessPainter._position!.dx.toInt(),
+            ChessPainter._position!.dy.toInt(), -1)) {
       TipsDialog.show(context, "很遗憾", "决策树算法打败了您");
       TipsDialog.show(context, "很遗憾", "决策树算法打败了您");
     }
     }
     setState(() {
     setState(() {
-      ChessPainter._position!.dx = ChessPainter._position!.dx*(width/15);
-      ChessPainter._position!.dy = ChessPainter._position!.dy*(width/15);
+      ChessPainter._position!.dx = ChessPainter._position!.dx * (width / 15);
+      ChessPainter._position!.dy = ChessPainter._position!.dy * (width / 15);
     });
     });
   }
   }
 }
 }
 
 
 class ChessPainter extends CustomPainter {
 class ChessPainter extends CustomPainter {
-  static List<Position> _positions = [];
   static int _state = 0;
   static int _state = 0;
   static Position? _position;
   static Position? _position;
   final Function _function;
   final Function _function;
+  Originator _originator = Originator.getInstance();
 
 
-  ChessPainter(Function f):_function = f;
+  ChessPainter(Function f) : _function = f;
 
 
   @override
   @override
   void paint(Canvas canvas, Size size) {
   void paint(Canvas canvas, Size size) {
@@ -141,8 +161,10 @@ class ChessPainter extends CustomPainter {
     var dx = _position!.dx;
     var dx = _position!.dx;
     var dy = _position!.dy;
     var dy = _position!.dy;
     for (int i = 0; i < CheckerBoardPainter._crossOverBeanList.length; i++) {
     for (int i = 0; i < CheckerBoardPainter._crossOverBeanList.length; i++) {
-      var absX = (dx - CheckerBoardPainter._crossOverBeanList[i]._dx).abs(); //两个点的x轴距离
-      var absY = (dy - CheckerBoardPainter._crossOverBeanList[i]._dy).abs(); //两个点的y轴距离
+      var absX =
+          (dx - CheckerBoardPainter._crossOverBeanList[i]._dx).abs(); //两个点的x轴距离
+      var absY =
+          (dy - CheckerBoardPainter._crossOverBeanList[i]._dy).abs(); //两个点的y轴距离
       var s = sqrt(absX * absX +
       var s = sqrt(absX * absX +
           absY * absY); //利用直角三角形求斜边公式(a的平方 + b的平方 = c的平方)来计算出两点间的距离
           absY * absY); //利用直角三角形求斜边公式(a的平方 + b的平方 = c的平方)来计算出两点间的距离
       if (s <= mWidth / 2 - 2) {
       if (s <= mWidth / 2 - 2) {
@@ -150,10 +172,11 @@ class ChessPainter extends CustomPainter {
         //找到离触摸点最近的棋盘坐标点并记录保存下来
         //找到离触摸点最近的棋盘坐标点并记录保存下来
         _position!.dx = CheckerBoardPainter._crossOverBeanList[i]._dx;
         _position!.dx = CheckerBoardPainter._crossOverBeanList[i]._dx;
         _position!.dy = CheckerBoardPainter._crossOverBeanList[i]._dy;
         _position!.dy = CheckerBoardPainter._crossOverBeanList[i]._dy;
-        _positions.add(_position!);
+        _originator.add(_position!);
         add = true;
         add = true;
         if (_position!.chess is WhiteChess) {
         if (_position!.chess is WhiteChess) {
-          Ai.getInstance().addChessman(_position!.dx~/(width/15), _position!.dy~/(width/15), 1);
+          Ai.getInstance().addChessman(
+              _position!.dx ~/ (width / 15), _position!.dy ~/ (width / 15), 1);
         }
         }
         // flag = false; //白子下完了,该黑子下了
         // flag = false; //白子下完了,该黑子下了
         break;
         break;
@@ -162,15 +185,25 @@ class ChessPainter extends CustomPainter {
 
 
     //画子
     //画子
     mPaint..style = PaintingStyle.fill;
     mPaint..style = PaintingStyle.fill;
-    if (_positions.isNotEmpty) {
-      for (int i = 0; i < _positions.length; i++) {
-        mPaint..color = _positions[i].chess.color;
-        canvas.drawCircle(Offset(_positions[i].dx, _positions[i].dy),
-            min(mWidth / 2, mHeight / 2) - 2, mPaint);
+    if (_originator.state.isNotEmpty) {
+      for (int i = 0; i < _originator.state.length; i++) {
+        mPaint..color = _originator.state[i].chess.color;
+        if (_originator.state[i].chess.chessShape.shape == 1) {
+          canvas.drawCircle(
+              Offset(_originator.state[i].dx, _originator.state[i].dy),
+              min(mWidth / 2, mHeight / 2) - 2,
+              mPaint);
+        }
+        if (_originator.state[i].chess.chessShape.shape == 2) {
+          Rect rect = Rect.fromCircle(
+              center: Offset(_originator.state[i].dx, _originator.state[i].dy),
+              radius: min(mWidth / 2, mHeight / 2) - 2);
+          canvas.drawRect(rect, mPaint);
+        }
       }
       }
     }
     }
     WidgetsBinding.instance!.addPostFrameCallback((_) {
     WidgetsBinding.instance!.addPostFrameCallback((_) {
-      if (add &&_position!.chess is WhiteChess) {
+      if (add && _position!.chess is WhiteChess) {
         _function();
         _function();
       }
       }
     });
     });
@@ -179,7 +212,6 @@ class ChessPainter extends CustomPainter {
   //在实际场景中正确利用此回调可以避免重绘开销,本示例我们简单的返回true
   //在实际场景中正确利用此回调可以避免重绘开销,本示例我们简单的返回true
   @override
   @override
   bool shouldRepaint(CustomPainter oldDelegate) {
   bool shouldRepaint(CustomPainter oldDelegate) {
-    // TODO: implement shouldRepaint
     return true;
     return true;
   }
   }
 }
 }
@@ -234,7 +266,6 @@ class CheckerBoardPainter extends CustomPainter {
   //在实际场景中正确利用此回调可以避免重绘开销,本示例我们简单的返回true
   //在实际场景中正确利用此回调可以避免重绘开销,本示例我们简单的返回true
   @override
   @override
   bool shouldRepaint(CustomPainter oldDelegate) {
   bool shouldRepaint(CustomPainter oldDelegate) {
-    // TODO: implement shouldRepaint
     return false;
     return false;
   }
   }
 }
 }

+ 2 - 1
lib/ai/Ai.dart

@@ -4,6 +4,7 @@ import 'dart:core';
 
 
 import 'dart:core';
 import 'dart:core';
 
 
+import 'package:gobang/bridge/ChessShape.dart';
 import 'package:gobang/flyweight/ChessFlyweightFactory.dart';
 import 'package:gobang/flyweight/ChessFlyweightFactory.dart';
 import 'package:gobang/flyweight/Position.dart';
 import 'package:gobang/flyweight/Position.dart';
 
 
@@ -310,7 +311,7 @@ class Ai{
     }
     }
     print(maxScore);
     print(maxScore);
 
 
-    if(goalX != -1 && goalY != -1){
+    if (goalX != -1 && goalY != -1) {
       return new Position(goalX.toDouble(), goalY.toDouble(), ChessFlyweightFactory.getInstance().getChess(""));
       return new Position(goalX.toDouble(), goalY.toDouble(), ChessFlyweightFactory.getInstance().getChess(""));
     }
     }
 
 

+ 17 - 0
lib/bridge/ChessShape.dart

@@ -0,0 +1,17 @@
+abstract class ChessShape{
+  int? _shape;
+
+  int get shape => _shape!;
+}
+
+class CircleShape extends ChessShape{
+  CircleShape(){
+    _shape = 1;
+  }
+}
+
+class RectShape extends ChessShape{
+  RectShape(){
+    _shape = 2;
+  }
+}

+ 16 - 0
lib/flyweight/Chess.dart

@@ -1,23 +1,39 @@
 import 'dart:ui';
 import 'dart:ui';
 
 
 import 'package:flutter/material.dart';
 import 'package:flutter/material.dart';
+import 'package:gobang/bridge/ChessShape.dart';
 
 
 /// 棋子的抽象类
 /// 棋子的抽象类
+/// 使用了桥接模式,外观和颜色是两个不同的维度
 abstract class Chess{
 abstract class Chess{
 
 
   Color? _color;
   Color? _color;
 
 
   Color get color => _color!;
   Color get color => _color!;
+
+  ChessShape? _chessShape;
+
+  ChessShape get chessShape => _chessShape!;
+
+  set chessShape(ChessShape? __chessShape);
 }
 }
 
 
 class BlackChess extends Chess{
 class BlackChess extends Chess{
   BlackChess() {
   BlackChess() {
     _color = Colors.black;
     _color = Colors.black;
   }
   }
+
+  set chessShape(ChessShape? __chessShape) {
+    super._chessShape = __chessShape;
+  }
 }
 }
 
 
 class WhiteChess extends Chess{
 class WhiteChess extends Chess{
   WhiteChess() {
   WhiteChess() {
     _color = Colors.white;
     _color = Colors.white;
   }
   }
+
+  set chessShape(ChessShape? __chessShape) {
+    super._chessShape = __chessShape;
+  }
 }
 }

+ 0 - 8
lib/flyweight/Decoration.dart

@@ -1,8 +0,0 @@
-import 'package:flutter/cupertino.dart';
-
-class Decoration{
-  Color? _color;
-  String? _shape;
-
-  Decoration(this._color, this._shape);
-}

+ 20 - 0
lib/memorandum/CareTaker.dart

@@ -0,0 +1,20 @@
+import 'Memo.dart';
+
+class CareTaker{
+  List<Memo> mementoList = [];
+
+  void add(Memo state){
+    mementoList.add(state);
+    if (mementoList.length>3) {
+      mementoList.removeAt(0);
+    }
+  }
+
+  Memo get(int index){
+    return mementoList[index];
+  }
+
+  Memo getLast(){
+    return mementoList.last;
+  }
+}

+ 11 - 0
lib/memorandum/Memo.dart

@@ -0,0 +1,11 @@
+import 'package:gobang/flyweight/Position.dart';
+
+class Memo {
+  List<Position> _state = [];
+
+  List<Position> get state => _state;
+
+  set state(List<Position> value) {
+    _state = value;
+  }
+}

+ 46 - 0
lib/memorandum/Originator.dart

@@ -0,0 +1,46 @@
+import 'package:gobang/flyweight/Position.dart';
+import 'package:gobang/memorandum/CareTaker.dart';
+
+import 'Memo.dart';
+
+class Originator {
+
+  Originator._();
+
+  static Originator? _originator;
+
+  static getInstance(){
+    if(_originator == null){
+      _originator = Originator._();
+    }
+    return _originator;
+  }
+
+  List<Position> _state = [];
+
+  List<Position> get state => _state;
+
+  CareTaker _careTaker = CareTaker();
+
+  add(Position position) {
+    _state.add(position);
+    _careTaker.add(_save());
+  }
+
+  _from(Memo memo) {
+    this._state = memo.state;
+  }
+
+  Memo _save() {
+    return Memo()..state = this._state;
+  }
+
+  clean(){
+    _state = [];
+  }
+
+  undo(){
+    Memo memo = _careTaker.getLast();
+    _from(memo);
+  }
+}

+ 88 - 0
lib/state/State.dart

@@ -0,0 +1,88 @@
+import 'package:gobang/state/UserContext.dart';
+
+abstract class State {
+  int _step = 0;
+  int _rect = 0;
+  UserContext _userContext;
+
+  State(UserContext userContext):_userContext = userContext;
+
+  // 悔棋只能悔棋三次
+  bool regretChess();
+
+  // 认输10步之内不能认输
+  bool surrender();
+
+  play() {
+    _step++;
+  }
+
+}
+
+/// [StartState] 开始状态
+class StartState extends State {
+
+  StartState(UserContext userContext) : super(userContext);
+
+
+
+  // 悔棋只能悔棋三次
+  @override
+  bool regretChess(){
+    return false;
+  }
+
+  @override
+  bool surrender() {
+    return false;
+  }
+
+  @override
+  play() {
+    super.play();
+    if(_step >= 10) {
+      _userContext.setState(MidState(_userContext).._step = _step.._rect = _rect);
+    }
+  }
+
+}
+
+/// [MidState] 中场状态
+class MidState extends State {
+  MidState(UserContext userContext) : super(userContext);
+
+  // 悔棋只能悔棋三次
+  @override
+  bool regretChess(){
+    if(_rect > 3) {
+      _userContext.setState(EndState(_userContext).._step = _step.._rect = _rect);
+      return false;
+    }
+    return true;
+  }
+
+  @override
+  bool surrender() {
+    return true;
+  }
+
+}
+
+/// [EndState] 结尾状态
+class EndState extends State {
+  EndState(UserContext userContext) : super(userContext);
+
+
+
+  // 悔棋只能悔棋三次
+  @override
+  regretChess(){
+    return false;
+  }
+
+  @override
+  surrender() {
+    return true;
+  }
+
+}

+ 28 - 0
lib/state/UserContext.dart

@@ -0,0 +1,28 @@
+import 'package:gobang/state/State.dart';
+
+class UserContext {
+
+  late State _state;
+
+  UserContext(){
+    _state = StartState(this);
+  }
+
+  play() {
+    _state.play();
+  }
+
+  // 悔棋只能悔棋三次
+  bool regretChess() {
+    return _state.regretChess();
+  }
+
+  // 认输10步之内不能认输
+  bool surrender() {
+    return _state.surrender();
+  }
+
+  setState(State state){
+    _state = state;
+  }
+}

+ 35 - 0
lib/viewModel/GameViewModel.dart

@@ -0,0 +1,35 @@
+import 'package:gobang/bridge/ChessShape.dart';
+import 'package:gobang/flyweight/Chess.dart';
+import 'package:gobang/flyweight/ChessFlyweightFactory.dart';
+import 'package:gobang/state/UserContext.dart';
+
+class GameViewModel {
+  GameViewModel._();
+
+  static GameViewModel? _gameViewModel;
+
+  static getInstance() {
+    if (_gameViewModel == null) {
+      _gameViewModel = GameViewModel._();
+    }
+    return _gameViewModel;
+  }
+
+  UserContext _userContext = UserContext();
+
+  Chess play() {
+    _userContext.play();
+    Chess chess;
+    /// 设置棋子外观
+    ChessShape shape = RectShape();
+    chess = ChessFlyweightFactory.getInstance().getChess("white");
+    chess.chessShape = shape;
+    return chess;
+  }
+
+  bool undo() {
+    return _userContext.regretChess();
+  }
+
+
+}