Skip to content

Latest commit

 

History

History
531 lines (396 loc) · 26.5 KB

README.md

File metadata and controls

531 lines (396 loc) · 26.5 KB

LDTS_T13_G1307 - CHESS

BCH compliance

  • TEST IN SPOCK: BetterCodeHub fails to recognize tests in spock.

  • Error-Prone: we have left only 4 warnings that we think are not worth of changing (1 related to the game loop, the others to the use of printstacktrace())

  • Pitest : we opt to omit some classes from the analysis which we thought were not susceptible of analysis

Game Description

This version of Chess is a representation of the classic strategy tabletop game Chess played between 2 players, in alternating turns. Each player has at their disposal 16 chess pieces (8 pawns, 2 rooks, 2 knights, 2 bishops, 1 queen and 1 king). Each type of piece has specific movement rules. When a player moves a piece to the same square as an enemy piece, the enemy piece is taken and the square is occupied by the player piece. The king piece has to stay alive at all points, so when a check is initiated, the player needs to act in order to protect their king. Once the king has no safe squares from a check, it is a check-mate and the player loses.

In the game, you will start in a menu in which you will have to choose the time control you want. The game provides all the standard rules and move behaviours of a chess game, you can't commit suicide neither make any move that's illegal, all moves and pieces are granted to be working as of a normal game. To make a move, you select with your cursor the piece you want to move and then the legal moves in red will indicate to which squares you can go. If you don't have any legal move then it's either a draw or, in case you are in check , a checkmate. If that's the case a pop-up screen will appear telling you the game has ended. You can lose by time too, there are two types of clocks available, clocks with increments and without them. In any time of the game you can press Backspace and go back to the main menu. In a check the king will turn purple in order to help you better understand what's going on. And last but not least, the game has sound making it more exiting then ever before.

This project was developed by Lucas Sousa ([email protected]), Vitor Cavaleiro ([email protected]) and Ricardo Matos ([email protected]) for LDTS 2021-22.

Implemented Features

  • Menu - When first running the game, the user will be met with a menu that can be navigated with the left and right arrows keys. It has multiple game modes and exit as options.
  • Multiple Game Modes - Each option in the menu corresponds to a different game mode, with a different clock duration. In some game modes the clock extends after every move.
  • 2 Player Game - The game will show a standard 8 by 8 square board, with 8 pawns, 2 rooks, 2 knights, 2 bishops, 1 queen and 1 king on either side, where 2 players take alternating turns moving their pieces.
  • Piece Movement - The pieces of the game, when selected will be able to be moved to a correct, legal position based on the rules of the game (Ex: knight will only be able to move in an L shape, the bishop will only be able to move diagonally). There are also situational moves such as castle and en passant.
  • Piece capturing - When a Piece is moved to a square occupied by an enemy piece, it will capture it. The enemy piece will be removed from the board and the other piece will take its square.
  • Cursor - In each turn, the player will be able to select the desired piece, with the arrow keys, and then select a legal square for the piece to move into.
  • Check - When, in the current round, an enemy piece is threatening to capture the king a check is initiated. When this happens the player must act in order to protect the king and stop the check. The king in check will be highlighted in purple.
  • Checkmate - During a check, when there is no possible way to protect the king and stop the check, there will be a Checkmate. The player who initiated the check will win and the game will return to the Main Menu.
  • Countdown Clock - During the game, each turn, the corresponding clock will start ticking down, stopping when the player has made their move. Subsequently, the other player's clock will do the same. When a player's clock reaches 0 the game will end and that player will lose. Clock duration is based on the game mode. In some game modes, such as Blitz, the clock duration increases after every move.
  • Sound - The game plays a sound for movements, captures and checkmate.
  • Rules - Illegal Moves are not allowed, rules restrains the movement of pieces. Note that the checkmate and stalemate flags are set by the rules of the game.

Design Decisions

As said above, the rules triggers the endGame events, they are defined by order of importance. They are also separated from the pieces which makes it cleaner and easier to add new rules and defining new games with it. The Moving behaviours of the pieces are also separated from the pieces per se. In this way we can re-utilize moving behaviour and combine them to make another ones.

General Structure

Problem in Context:

In order to make the code more reusable , organizable , and so that it wouldn't break the Single Principle Responsibility (which could easily happen) we decided to use the MVC Architecture Pattern .

The Pattern:

The MVC pattern separates the code in three packages the Model, View and Controller. The model represents the logical part of the program, the View is responsible for showing the program (depending on the model) and the controller is responsible for controlling the states of the program (depending on the previous two).

Implementation:

  • Model - Stores the data pertaining to the game, such as the positions of the pieces and their possible movements, as well as the logic of the game and every piece.
  • View - Handles the output of the graphical interface of the game through Lanterna, drawing the menu, board and its pieces.
  • Controller - Controls the flow of the program.


Fig 1. Model, Controller and Viewer pattern design



Consequences:

  • Organization of source code, allowing for better development
  • Modifications don't affect the entire program
  • Better separation of the program logic from the view drawing
  • Easier addition of features during development.
  • Single Responsibility Principle is not violated


Moving Pieces

Problem in context:

Moving a piece to a new square should be straight forward. You just grab a piece and put it in a new square. However, there are moves that require a little more. Capturing, for instance, requires that the captured piece is eliminated from the pieces in game, and then the piece that is capturing can go and take the new square. Other example would be castling or promoting, in which, apart from the usual move of moving the piece to the new square, we have to make others changes too (the rook should change column and the pawn should be promoted to another piece). So we opt to use the decorator pattern, which will allow us to not violate the Single Responsibility Principle , as we could have put all these behaviours in a class that implemented all of them, and let us create new move behaviours that can be combined into new ones making the code cleaner and reusable (we mainly used the Simple Move as the wrappee however we could have made funny things like, when you capture a piece you are able to promote it to another , this happens because this pattern is good for combining behaviours by wrapping the objets into multiple decorators )

The Pattern:

We have applied the decorator pattern. Using it, we can extend a move behaviour combining it with multiple ones that are wrapped inside it. A very good example of this pattern in action is when we capture a piece with a pawn in the 8 th rank, when that happens a call to the promoting move is made, followed by a call to the capturing move follow by the simple move call.

Implementation:

The interface Move defines the 3 methods that every Move must have, execute method is responsible for making the move, while the others 2 are responsible for getting information about the moving piece and his new position. The MoveDecorator is responsible for wrapping the Moves. Capturing, Castling and Promoting (the last two are private classes in the rules responsible for handling/creating them) are moves that add an extra behaviour to the SimpleMove. That way, the SimpleMove (concrete component) defines the basic behaviour altered (by addition) from the concrete Decorators (CapturingMove...).



Fig 2. Decorator pattern



These classes can be found in the following files:

Consequences:

Benefits of applying the above pattern:

  • Combination of several move behaviours by wrapping them into multiple decorators
  • Adding extra behaviours to an object at runtime
  • Single Responsibility Principle obeyed


Board Strategy (2x, with the Clock)

Problem in context:

Even though we only planned on the game having only one kind of board, the classic 8x8 board, we decided to develop it in a way that would allow for multiple types of boards. In this way we protected ourselves from going against the Open-Closed Principle and we let the game open for new variants of the game. We actually used this principle more than once , in the Clock Package we use it so that one could create either a more precise clock or a clock with more functionalities. This seems unnecessary, but it actually led us to have less concern between modules and fewer worryings about what a concrete class was doing and how it was doing .

The Pattern:

We have applied the Strategy Pattern. This way we prevent future violations of the SOLID principles. It's also a good practise, by not "working" for the implementation but for the interface.

Implementation:

The interface BoarModel defines two methods that are intrinsic to every board. A board should be able to tell if a position is inside or outside it, and it also should be able to tell which are those cases. The last one is crucial as it is essential for the rendering of the board. In the clock something similar happens.

Fig 3. BoardModel Strategy (only SquareBoard is implemented)



These classes can be found in the following files:

Consequences:

Benefits of applying the above pattern:

  • Allows for different types of boards that differ from the classical square board.
  • Isolate the implementation details.
  • Obeys Open/Closed Principle. New strategies can be introduced without having to change the context.


Possible moves for different pieces

Problem in context:

Each type of piece in the game can only move in very specific ways (Ex. Bishop can only move to diagonal squares), and some pieces move in similar ways to other pieces (Ex. Queen can move like a Bishop and like a Rook).

The pattern:

We have applied the Composite Pattern. This way we can create general behaviours for the more common moves, such as moving side-to-side or moving diagonally, and more specific behaviours for pieces like the Pawn. We also have moving behaviour groups that combine different behaviours (Ex. The MovingBehaviourGroup associated with the queen is made up of a side moving behaviour and an adjacent moving behaviour). Our implementation however, does not provide the option of recursively iterate through moving behaviours (we used a set here) as it wouldn't make sense to have a moving behaviour group with more than one equal moving behaviour (those squares would already be considered from the previous moving behaviour), so one could say that this is more of a Strategy Pattern that calls combined strategy's . This pattern allows us to obey the Open/Closed Principle as it lets us introduce new Moving behaviours without altering pre-existing code.

Implementation:

Fig 4. Composite Pattern in MovingBehaviours



These classes can be found in the following files:

Consequences:

Benefits of applying the above pattern:

  • Allows for simpler moving behaviours that can be composed into the more complex behaviours of the pieces.
  • Allows for the implementation of movingBehaviours that can be utilized by multiple pieces.
  • Isolate the implementation details.
  • Obeys Open/Closed Principle. New moving behaviours can be introduced without having to change the context.


Player's turn

Problem in context:

After every move the players must be informed if it is their turn to play. If it is, then the player clock must start ticking. So, the player must observe the board so that it can know when the game turn state has been changed .

The Pattern:

We have applied the Observer Pattern . With it, we can notify the player when it is his turn to play.

Implementation:

The notifyPlayers method in GameModel will inform the gamePlayers when a move is made and if it is their turn, the responsibility to make the clock ticking/ stopping is of the Player. Every player implements an interface called GameSubscriber that tells the player if it is their turn, informing the player that a move was made.



Fig 5. Observer Pattern



These classes can be found in the following files:

Consequences:

Benefits of applying the above pattern:

  • Follows Open/Close Principle.
  • Establishes relationships between objects allowing the players to know the state of the game.


Building the chess game

Problem in context:

Before a game can start, there needs to be a defined set of rules, a board and a set of pieces each player can use. The program structure allows the programmer to build a customized game, however, we will focus on the traditional chess game, for now. The fact that we construct a game based on pre-defined steps allows for the construction of games based on the clients will, we could for instance create a game object without rules ( which is known in chess as analysis).

The pattern:

We have applied the Builder Pattern. With this pattern, we can organize the construction of the game components in a much cleaner and organized way. It also allows the programmer to easily add different gameplay components. This way, we let the Game builder be the constructor of the game taking that responsibility from the GameModel, avoiding per se a code smell and respecting Single Responsibility Principle .

Implementation:

The standardChessGame class, the class that builds the classical chess game, has a build method for every type of piece, take the buildBishops method for example, it will add to both sets of pieces 2 bishops. The buildPieces method combines all the methods that create the different pieces and the updateModelPieces method applies the pieces to the gameModel. The buildRules method assigns to the gameModel the set of rules the game obeys. The getResults method returns the complete gameModel, which will be received by the gameState.

Fig 6. Builder Pattern



These classes can be found in the following files:

Consequences:

Benefits of applying the above pattern:

  • Respects the Single Responsibility Principle.
  • Allows for construction code to be reused when building different types of chess games.
  • Allows for the creation of objects step-by-step.


Applying the rules of the game

Problem in context:

Every chess game and every variant of it has rules. We could have applied those rules in the pieces moving behaviour or even in the pieces , however, this would come at a great price, the violation of Single Responsibility Principle. Why? Chess pieces would do more than they should and also know more than they should ( and this could become a future Open/Closed Principle violation). So we opted to use the Filter Pattern which solves the previous issues and gives the flexibility to add/remove rules based on the clients wish without implying a SOLID violation.

The pattern:

We opted to use the Filter pattern. This pattern is responsible for applying the rules based on the criteria defined by it. Having the liberty to make changes to the moves that a piece can make without letting it know about those rules.

Implementation:

Every class that implements a rule must receive a piece and a set of moves and apply the criteria defined by it.



Fig 7. Rule Pattern



These classes can be found in the following package:

Consequences:

Benefits of applying the above pattern:

  • Allows for more flexible code making it cleaner and easier.
  • Allows for the introduction of new rules without breaking Open/Closed Principle.
  • Keeps the Responsibility of dealing with rules separate from the Pieces (Single Principle Responsibility).


Controller operation

Problem in context:

Since the program follows the MVC pattern, there needs to be a controller that reads the user inputs and manages the flow of the game.

The pattern:

The implemented pattern is the State Pattern. It behaves like a finite-state machine, meaning there is a number of finite states, and each transition, from one state to another, must follow a rule made by the programmer.

Implementation:

When the controller is initialized and executed, the MenuState is set to the default state of the controller. Depending on the user input, the MenuState will determine the next state and so on... When the user chooses the exit option, the state is set to null and then the program closes. When the user chooses the new game option the closeIfMoving method closes the menu window and the game window opens. The game state behaves similarly, changing for the game end state when the game ends.



Fig 8. State Pattern



These classes can be found in the following files:

Consequences:

Benefits of applying the above pattern:

  • Simplifies the code.
  • Follows the Open/Closed Principle.
  • Follows the Single Responsibility Principle.


Class Diagram



Fig 9. Class UML



Refactoring list

  • Reduced the number of method calls on rule type classes, separating concerns in modules
  • Eliminated similar clear methods on the view type classes(pull-up method).
  • Eliminated duplicate methods on the piece type classes(pull-up method).
  • Substitute Algorithm on moving behaviours, in fact we did this a lot of times
  • Add Parameter , we did this a few times specially when using dependency injection. We did this in the Rule interface too.
  • Push up Field We had to this because of point 3.
  • ...

Known Code Smells And Refactoring Suggestions

Large Class

We consider the StandardChessGame (128 lines) class a Large Class due to having a builder method for every piece. However, we find justifiable since every piece has an unique starting position in the layout of the game and in this way the code gets cleaner and easier to understand ( we could have combined all the methods into one big one but that is not a good principle)

We also see the GameModel (157 lines) as Large Class due to the amount of lines that it has . We could have extracted the Flow of the game to a new class or subclass but then GameModel would become a Data class. Also , as said before, rules are responsible for "setting flags" that then the game interprets and sets the GameEvent (a checkmate occurs after all moves are made illegal by rules and the king is in check), so there would be no need for GameEventClass that determine the game flow.

The Two classes are responsible for one thing, so this is not "a big problem". But, if we were to add more code this classes would probably needed to be refactored.

Lazy Class

We consider some child classes of the Piece class (Ex. Queen, Rook, etc) to be Lazy Classes. This is due to the fact that they don't do much outside of attributing the correct moving behaviour group and the character that corresponds with the symbol in the font that's shown on screen. He could have used Collapse Hierarchy and set the pieces as a concrete implementation of Piece (maybe even using a factory), but that wouldn't be possible to some special Pieces (Pawns and Kings) as they need to be able to overload some operators.

We also believe the LStrategy and AdjacentStrategy classes to be lazy classes as they are just a Constructor of the abstract method. This was actually a refactor that we did, it allows us to not fall in another code smell, Duplicated code .

Data Class

We consider the Player class a Data Class, considering that, besides telling the clock to pause or resume, it serves only to store the data about the clock and the color of the pieces. It actually operates once in his data, but is not independently . He could delete it but then the game itself would become warder to understand (a game without a player sounds weird).

We also count BoardCase as a Data Class, due to the fact that it is a record .

We find them justifiable as they increase the organization of the code and made development more intuitive.

Switch Case

The GameState and MenuState classes contain a switch case code smell each.

In GameState, the large switch case serves to process the user input during the game, managing the due Controller State.

The switch case in MenuState works in a simillar fashion to the previous one when it comes to navigating the menu. It also starts the correct GameState based on the chosen game mode.

Considering these use cases, we find the switch cases to be acceptable as they perform simple actions .

Feature Envy

The Feature Envy smell is present in the Rule classes (with the Rule interface). These classes decide which of possible moves for a piece are legal, filtering their possible moves. As such, they access the data of the pieces more than their own. Due to our implementation of the pieces and behaviours, we find this code smell justifiable. Note that this smell comes from the use of the filter pattern .



Testing

Screenshot of coverage report

Fig 10. Code coverage screenshot

Mutation Testing Report

Fig 11. Pit Test Coverage Report

Link to Pit Test Mutation report

Self-evaluation

  • Vitor Cavaleiro: 30%
  • Ricardo de Matos: 50%
  • Lucas Sousa: 20%