Merge branch 'develop'
-Correct bugs -Manage score -Add some helpers -Optimize some code
This commit is contained in:
commit
8488827025
8 changed files with 353 additions and 177 deletions
|
@ -2,14 +2,22 @@
|
||||||
#include <SFML/Window/Keyboard.hpp>
|
#include <SFML/Window/Keyboard.hpp>
|
||||||
#include "../../Helpers/Keyboard.hpp"
|
#include "../../Helpers/Keyboard.hpp"
|
||||||
|
|
||||||
|
|
||||||
|
//==================== Constructor and Destructor ====================
|
||||||
|
|
||||||
|
//Constructor
|
||||||
ConsoleController::ConsoleController()
|
ConsoleController::ConsoleController()
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//Destructor
|
||||||
ConsoleController::~ConsoleController()
|
ConsoleController::~ConsoleController()
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//==================== Helpers ====================
|
||||||
|
|
||||||
|
//Run the game.
|
||||||
void ConsoleController::run()
|
void ConsoleController::run()
|
||||||
{
|
{
|
||||||
|
|
||||||
|
@ -22,6 +30,9 @@ void ConsoleController::run()
|
||||||
//Pop a random number on the grid
|
//Pop a random number on the grid
|
||||||
m_game.popRandomNumber();
|
m_game.popRandomNumber();
|
||||||
|
|
||||||
|
//First cout stats
|
||||||
|
this->coutStats();
|
||||||
|
|
||||||
//First cout grid
|
//First cout grid
|
||||||
m_game.coutGrid();
|
m_game.coutGrid();
|
||||||
|
|
||||||
|
@ -33,20 +44,27 @@ void ConsoleController::run()
|
||||||
keyPress=this->waitArrowKeyPress();
|
keyPress=this->waitArrowKeyPress();
|
||||||
|
|
||||||
//Apply move
|
//Apply move
|
||||||
m_game.swipe(keyPress);
|
bool moveDone=m_game.swipe(keyPress);
|
||||||
|
|
||||||
//Pop a random number on the grid
|
//Cout stats
|
||||||
m_game.popRandomNumber();
|
this->coutStats();
|
||||||
|
|
||||||
//Cout grid
|
//Cout grid
|
||||||
m_game.coutGrid();
|
m_game.coutGrid();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//Last cout stats
|
||||||
|
this->coutStats();
|
||||||
|
|
||||||
|
//Last cout grid
|
||||||
m_game.coutGrid();
|
m_game.coutGrid();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
//Wait for keypress and return the keyPress.
|
||||||
kbdh::Direction ConsoleController::waitArrowKeyPress()
|
kbdh::Direction ConsoleController::waitArrowKeyPress()
|
||||||
{
|
{
|
||||||
//Initialise keyPress
|
//Initialise keyPress
|
||||||
|
@ -59,7 +77,8 @@ kbdh::Direction ConsoleController::waitArrowKeyPress()
|
||||||
keyPress=kbdh::Left;
|
keyPress=kbdh::Left;
|
||||||
while(sf::Keyboard::isKeyPressed(sf::Keyboard::Left))
|
while(sf::Keyboard::isKeyPressed(sf::Keyboard::Left))
|
||||||
{
|
{
|
||||||
//Wait for release
|
//Wait for release and try to remove arrow printed characters
|
||||||
|
std::cout << "\r" << " ";
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -68,7 +87,8 @@ kbdh::Direction ConsoleController::waitArrowKeyPress()
|
||||||
keyPress=kbdh::Right;
|
keyPress=kbdh::Right;
|
||||||
while(sf::Keyboard::isKeyPressed(sf::Keyboard::Right))
|
while(sf::Keyboard::isKeyPressed(sf::Keyboard::Right))
|
||||||
{
|
{
|
||||||
//Wait for release
|
//Wait for release and try to remove arrow printed characters
|
||||||
|
std::cout << "\r" << " ";
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -77,17 +97,18 @@ kbdh::Direction ConsoleController::waitArrowKeyPress()
|
||||||
keyPress=kbdh::Up;
|
keyPress=kbdh::Up;
|
||||||
while(sf::Keyboard::isKeyPressed(sf::Keyboard::Up))
|
while(sf::Keyboard::isKeyPressed(sf::Keyboard::Up))
|
||||||
{
|
{
|
||||||
//Wait for release
|
//Wait for release and try to remove arrow printed characters
|
||||||
|
std::cout << "\r" << " ";
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if (sf::Keyboard::isKeyPressed(sf::Keyboard::Down))
|
if (sf::Keyboard::isKeyPressed(sf::Keyboard::Down))
|
||||||
{
|
{
|
||||||
// la touche "flèche gauche" est enfoncée : on bouge le personnage
|
|
||||||
keyPress=kbdh::Down;
|
keyPress=kbdh::Down;
|
||||||
while(sf::Keyboard::isKeyPressed(sf::Keyboard::Down))
|
while(sf::Keyboard::isKeyPressed(sf::Keyboard::Down))
|
||||||
{
|
{
|
||||||
//Wait for release
|
//Wait for release and try to remove arrow printed characters
|
||||||
|
std::cout << "\r" << " ";
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -95,3 +116,11 @@ kbdh::Direction ConsoleController::waitArrowKeyPress()
|
||||||
|
|
||||||
return keyPress;
|
return keyPress;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//Cout the stats of the game
|
||||||
|
void ConsoleController::coutStats(){
|
||||||
|
|
||||||
|
std::cout << std::endl << "Score : " << m_game.getScore() << std::endl;
|
||||||
|
std::cout << "Nombre de coups : " << m_game.getNbMove() << std::endl;
|
||||||
|
}
|
||||||
|
|
|
@ -18,9 +18,13 @@ class ConsoleController
|
||||||
Game m_game;
|
Game m_game;
|
||||||
kbdh::Direction waitArrowKeyPress();
|
kbdh::Direction waitArrowKeyPress();
|
||||||
public:
|
public:
|
||||||
|
//Constructor and Destructor
|
||||||
ConsoleController();
|
ConsoleController();
|
||||||
~ConsoleController();
|
~ConsoleController();
|
||||||
|
|
||||||
|
//Helpers
|
||||||
void run();
|
void run();
|
||||||
|
void coutStats();
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -10,6 +10,7 @@ namespace kbdh {
|
||||||
|
|
||||||
//Key arrow
|
//Key arrow
|
||||||
enum Direction { Up, Down, Left, Right };
|
enum Direction { Up, Down, Left, Right };
|
||||||
|
//Key arrow typedef
|
||||||
typedef enum Direction Direction;
|
typedef enum Direction Direction;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,49 +1,85 @@
|
||||||
#include "Game.hpp"
|
#include "Game.hpp"
|
||||||
|
|
||||||
|
|
||||||
|
//==================== Constructor and Destructor ====================
|
||||||
|
|
||||||
Game::Game() : m_grid(){
|
//Constructor
|
||||||
|
Game::Game() : m_grid(), m_score(0), m_nbMove(0){
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//Destructor
|
||||||
Game::~Game(){
|
Game::~Game(){
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//==================== Helpers ====================
|
||||||
|
|
||||||
|
//Swipe action
|
||||||
bool Game::swipe(kbdh::Direction direction){
|
bool Game::swipe(kbdh::Direction direction){
|
||||||
|
|
||||||
|
bool moveDone;
|
||||||
|
|
||||||
switch(direction){
|
switch(direction){
|
||||||
|
|
||||||
case kbdh::Left:
|
case kbdh::Left:
|
||||||
m_grid.swipeLeft();
|
moveDone=m_grid.swipeLeft();
|
||||||
break;
|
break;
|
||||||
case kbdh::Right:
|
case kbdh::Right:
|
||||||
m_grid.swipeRight();
|
moveDone=m_grid.swipeRight();
|
||||||
break;
|
break;
|
||||||
case kbdh::Up:
|
case kbdh::Up:
|
||||||
m_grid.swipeUp();
|
moveDone=m_grid.swipeUp();
|
||||||
break;
|
break;
|
||||||
case kbdh::Down:
|
case kbdh::Down:
|
||||||
m_grid.swipeDown();
|
moveDone=m_grid.swipeDown();
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
if(moveDone){
|
||||||
|
m_score+=m_grid.getLastMoveScore();
|
||||||
|
m_nbMove++;
|
||||||
|
this->popRandomNumber();
|
||||||
|
}
|
||||||
|
|
||||||
|
return moveDone;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//Cout the grid
|
||||||
void Game::coutGrid(){
|
void Game::coutGrid(){
|
||||||
std::cout << m_grid.description();
|
std::cout << m_grid.description();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//Return true if the game is lost. False else.
|
||||||
bool Game::isOver(){
|
bool Game::isOver(){
|
||||||
return m_grid.isOver();
|
return m_grid.isOver();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//Pop a random number on the grid
|
||||||
void Game::popRandomNumber(){
|
void Game::popRandomNumber(){
|
||||||
std::tuple<int, int> coord(m_grid.getRandomEmptyCellCoord());
|
std::tuple<int, int> coord(m_grid.getRandomEmptyCellCoord());
|
||||||
|
|
||||||
int number=2;
|
int percent=rand() % 100;
|
||||||
|
|
||||||
|
int number;
|
||||||
|
|
||||||
|
if(percent <= 10){
|
||||||
|
number=4;
|
||||||
|
}
|
||||||
|
else{
|
||||||
|
number=2;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
m_grid.setCell(coord, number);
|
m_grid.setCell(coord, number);
|
||||||
}
|
}
|
||||||
|
//==================== Getters and Setter ====================
|
||||||
|
|
||||||
|
//Retrieve the Score
|
||||||
|
int Game::getScore(){
|
||||||
|
return m_score;
|
||||||
|
}
|
||||||
|
|
||||||
|
//Retrieve the number of moves
|
||||||
|
int Game::getNbMove(){
|
||||||
|
return m_nbMove;
|
||||||
|
}
|
||||||
|
|
|
@ -16,16 +16,25 @@
|
||||||
class Game
|
class Game
|
||||||
{
|
{
|
||||||
private:
|
private:
|
||||||
|
//Members
|
||||||
Grid m_grid;
|
Grid m_grid;
|
||||||
|
int m_score;
|
||||||
|
int m_nbMove;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
//Constructor and Destructor
|
||||||
Game();
|
Game();
|
||||||
~Game();
|
~Game();
|
||||||
|
|
||||||
|
//Helpers
|
||||||
bool swipe(kbdh::Direction direction);
|
bool swipe(kbdh::Direction direction);
|
||||||
void coutGrid();
|
void coutGrid();
|
||||||
void popRandomNumber();
|
void popRandomNumber();
|
||||||
bool isOver();
|
bool isOver();
|
||||||
|
|
||||||
|
//Getters and Setters
|
||||||
|
int getScore();
|
||||||
|
int getNbMove();
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -1,5 +1,7 @@
|
||||||
#include "Grid.hpp"
|
#include "Grid.hpp"
|
||||||
|
|
||||||
|
//==================== Constructor and Destructor ====================
|
||||||
|
|
||||||
//Constructor
|
//Constructor
|
||||||
Grid::Grid(): m_size(4), m_grid(4){
|
Grid::Grid(): m_size(4), m_grid(4){
|
||||||
|
|
||||||
|
@ -9,7 +11,6 @@ Grid::Grid(): m_size(4), m_grid(4){
|
||||||
m_grid.at(i).push_back(0);
|
m_grid.at(i).push_back(0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//Destructor
|
//Destructor
|
||||||
|
@ -17,41 +18,140 @@ Grid::~Grid(){
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
std::string Grid::description(){
|
//==================== Merge and defragment methods ====================
|
||||||
|
|
||||||
//Init stringstream description
|
//Defragment the line to the right
|
||||||
std::stringstream description;
|
std::vector<int> Grid::rightDefragment(std::vector<int> line){
|
||||||
|
for(int j=0; j<m_size-1;j++){
|
||||||
|
for(int i=0; i<m_size-1;i++){
|
||||||
|
int val1=line.at(i);
|
||||||
|
int val2=line.at(i+1);
|
||||||
|
|
||||||
//Get max str len of the grid
|
if(val1 != 0 && val2 == 0){
|
||||||
int maxStrLen=this->maxStrLenInGrid();
|
line.at(i)=0;
|
||||||
|
line.at(i+1)=val1;
|
||||||
//Start to write description
|
|
||||||
std::stringstream gridBorder;
|
|
||||||
for(int i=0;i<(maxStrLen+2)*4+1;i++){
|
|
||||||
gridBorder<<"-";
|
|
||||||
}
|
|
||||||
description << std::endl << gridBorder.str() << std::endl;
|
|
||||||
for(int i=0;i<m_size;i++){
|
|
||||||
for(int j=0;j<m_size;j++){
|
|
||||||
std::stringstream spaceCol;
|
|
||||||
for(int k=0;k<maxStrLen-std::to_string(m_grid.at(i).at(j)).size();k++){
|
|
||||||
spaceCol << " ";
|
|
||||||
}
|
}
|
||||||
if(m_grid.at(i).at(j) == 0)
|
|
||||||
description << "| " << " " << spaceCol.str();
|
|
||||||
else
|
|
||||||
description << "| " << m_grid.at(i).at(j) << spaceCol.str();
|
|
||||||
}
|
}
|
||||||
description << "|";
|
|
||||||
description << std::endl;
|
|
||||||
}
|
}
|
||||||
description << gridBorder.str() << std::endl << std::endl;
|
return line;
|
||||||
|
}
|
||||||
|
|
||||||
//Return description
|
//Defragment the line to the left using right defragmentation
|
||||||
return description.str();
|
std::vector<int> Grid::leftDefragment(std::vector<int> line){
|
||||||
|
|
||||||
|
std::vector<int> reversedLine= this->reverseLine(line);
|
||||||
|
|
||||||
|
return this->reverseLine(this->rightDefragment(reversedLine));
|
||||||
|
}
|
||||||
|
|
||||||
|
//Merge line to the right
|
||||||
|
std::vector<int> Grid::rightMerge(std::vector<int> line){
|
||||||
|
for(int i=0; i< m_size-1;i++){
|
||||||
|
int val1=line.at(i);
|
||||||
|
int val2=line.at(i+1);
|
||||||
|
|
||||||
|
if(val1==val2){
|
||||||
|
line.at(i)=0;
|
||||||
|
line.at(i+1)=val1*2;
|
||||||
|
m_lastMoveScore+=val1*2;
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return line;
|
||||||
|
}
|
||||||
|
|
||||||
|
//Merge line to the left
|
||||||
|
std::vector<int> Grid::leftMerge(std::vector<int> line){
|
||||||
|
for(int i=m_size-1; i>0;i--){
|
||||||
|
int val1=line.at(i);
|
||||||
|
int val2=line.at(i-1);
|
||||||
|
|
||||||
|
if(val1==val2){
|
||||||
|
line.at(i)=0;
|
||||||
|
line.at(i-1)=val1*2;
|
||||||
|
m_lastMoveScore+=val1*2;
|
||||||
|
i--;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return line;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//==================== Swipe methods ====================
|
||||||
|
|
||||||
|
|
||||||
|
//Swipe to the right
|
||||||
|
bool Grid::swipeRight(){
|
||||||
|
|
||||||
|
m_lastMoveScore=0;
|
||||||
|
bool moveDone=false;
|
||||||
|
for(int i=0; i<m_size;i++){
|
||||||
|
std::vector<int> swipedLine(this->rightDefragment(this->leftMerge(this->rightDefragment(m_grid.at(i)))));
|
||||||
|
|
||||||
|
if(!this->compareLines(m_grid.at(i), swipedLine)){
|
||||||
|
moveDone=true;
|
||||||
|
m_grid.at(i)=swipedLine;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return moveDone;
|
||||||
|
}
|
||||||
|
|
||||||
|
//Swipe to the left
|
||||||
|
bool Grid::swipeLeft(){
|
||||||
|
m_lastMoveScore=0;
|
||||||
|
bool moveDone=false;
|
||||||
|
for(int i=0; i<m_size;i++){
|
||||||
|
std::vector<int> swipedLine(this->leftDefragment(this->rightMerge(this->leftDefragment(m_grid.at(i)))));
|
||||||
|
if(!this->compareLines(m_grid.at(i), swipedLine)){
|
||||||
|
moveDone=true;
|
||||||
|
m_grid.at(i)=swipedLine;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return moveDone;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//Swipe to the top
|
||||||
|
bool Grid::swipeUp(){
|
||||||
|
m_lastMoveScore=0;
|
||||||
|
bool moveDone=false;
|
||||||
|
for(int i=0; i<m_size;i++){
|
||||||
|
std::vector<int> colVect=this->getCol(i);
|
||||||
|
|
||||||
|
std::vector<int> swipedLine(this->leftDefragment(this->rightMerge(this->leftDefragment(colVect))));
|
||||||
|
if(!this->compareLines(colVect, swipedLine)){
|
||||||
|
moveDone=true;
|
||||||
|
this->setCol(i,swipedLine);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return moveDone;
|
||||||
|
}
|
||||||
|
//Swipe to the bottom
|
||||||
|
bool Grid::swipeDown(){
|
||||||
|
m_lastMoveScore=0;
|
||||||
|
bool moveDone=false;
|
||||||
|
for(int i=0; i<m_size;i++){
|
||||||
|
std::vector<int> colVect=this->getCol(i);
|
||||||
|
|
||||||
|
std::vector<int> swipedLine(this->rightDefragment(this->leftMerge(this->rightDefragment(colVect))));
|
||||||
|
|
||||||
|
if(!this->compareLines(colVect, swipedLine)){
|
||||||
|
moveDone=true;
|
||||||
|
this->setCol(i,swipedLine);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return moveDone;
|
||||||
|
}
|
||||||
|
|
||||||
|
//==================== Helpers ====================
|
||||||
|
|
||||||
|
//Get the max len of a string of the numbers in the grid :
|
||||||
|
// Exemple:
|
||||||
|
// - 1 return 1
|
||||||
|
// - 22 return 2
|
||||||
|
// - 120 return 3
|
||||||
|
// - 1000 return 4
|
||||||
|
//This method help to calculate the columns size
|
||||||
int Grid::maxStrLenInGrid(){
|
int Grid::maxStrLenInGrid(){
|
||||||
int max=0;
|
int max=0;
|
||||||
for(int i=0;i<m_size;i++){
|
for(int i=0;i<m_size;i++){
|
||||||
|
@ -64,12 +164,14 @@ int Grid::maxStrLenInGrid(){
|
||||||
return max;
|
return max;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//Test if the cell at (i,j) is empty
|
||||||
bool Grid::isEmpty(int i, int j){
|
bool Grid::isEmpty(int i, int j){
|
||||||
if(m_grid.at(i).at(j) == 0)
|
if(m_grid.at(i).at(j) == 0)
|
||||||
return true;
|
return true;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//Return a tuple that contain a random empty cell
|
||||||
std::tuple<int, int> Grid::getRandomEmptyCellCoord(){
|
std::tuple<int, int> Grid::getRandomEmptyCellCoord(){
|
||||||
|
|
||||||
//Init list of candidate
|
//Init list of candidate
|
||||||
|
@ -97,117 +199,7 @@ std::tuple<int, int> Grid::getRandomEmptyCellCoord(){
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//Reverse a line
|
||||||
|
|
||||||
//Change value of cell
|
|
||||||
bool Grid::setCell(std::tuple<int, int> coord, int value){
|
|
||||||
int i=std::get<0>(coord);
|
|
||||||
int j=std::get<1>(coord);
|
|
||||||
|
|
||||||
if(i>=0 && i<m_size && j>=0 && j<m_size){
|
|
||||||
m_grid.at(i).at(j)=value;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
//Another setCell method
|
|
||||||
bool Grid::setCell(int i, int j, int value){
|
|
||||||
std::tuple<int, int> coord(i,j);
|
|
||||||
return this->setCell(coord, value);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
std::vector<int> Grid::defragmentLine(std::vector<int> line){
|
|
||||||
for(int j=0; j<m_size-1;j++){
|
|
||||||
for(int i=0; i<m_size-1;i++){
|
|
||||||
int val1=line.at(i);
|
|
||||||
int val2=line.at(i+1);
|
|
||||||
|
|
||||||
if(val1 != 0 && val2 == 0){
|
|
||||||
line.at(i)=0;
|
|
||||||
line.at(i+1)=val1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return line;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::vector<int> Grid::mergeLine(std::vector<int> line){
|
|
||||||
for(int i=0; i< m_size-1;i++){
|
|
||||||
int val1=line.at(i);
|
|
||||||
int val2=line.at(i+1);
|
|
||||||
|
|
||||||
if(val1==val2){
|
|
||||||
line.at(i)=0;
|
|
||||||
line.at(i+1)=val1*2;
|
|
||||||
i++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return line;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::vector<int> Grid::swipeLine(std::vector<int> line){
|
|
||||||
|
|
||||||
//Swipe line is :
|
|
||||||
//- A defragmentation
|
|
||||||
//- A merging
|
|
||||||
//- Another defragmentation
|
|
||||||
line=this->defragmentLine(line);
|
|
||||||
line=this->mergeLine(line);
|
|
||||||
line=this->defragmentLine(line);
|
|
||||||
|
|
||||||
//Return swiped line
|
|
||||||
return line;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
//Swipe to right
|
|
||||||
void Grid::swipeRight(){
|
|
||||||
for(int i=0; i<m_size;i++){
|
|
||||||
m_grid.at(i)=this->swipeLine(m_grid.at(i));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//Swipe to right
|
|
||||||
void Grid::swipeLeft(){
|
|
||||||
for(int i=0; i<m_size;i++){
|
|
||||||
m_grid.at(i)=this->reverseLine(this->swipeLine(this->reverseLine(m_grid.at(i))));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void Grid::swipeUp(){
|
|
||||||
for(int i=0; i<m_size;i++){
|
|
||||||
std::vector<int> colVect=this->getCol(i);
|
|
||||||
this->setCol(i,this->reverseLine(this->swipeLine(this->reverseLine(colVect))));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
void Grid::swipeDown(){
|
|
||||||
for(int i=0; i<m_size;i++){
|
|
||||||
std::vector<int> colVect=this->getCol(i);
|
|
||||||
this->setCol(i,this->swipeLine(colVect));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void Grid::setCol(int col, std::vector<int> colVect){
|
|
||||||
for(int i=0;i<m_size;i++){
|
|
||||||
m_grid.at(i).at(col)=colVect.at(i);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
std::vector<int> Grid::getCol(int col){
|
|
||||||
|
|
||||||
std::vector<int> colVect;
|
|
||||||
|
|
||||||
for(int i=0;i<m_size;i++){
|
|
||||||
colVect.push_back(m_grid.at(i).at(col));
|
|
||||||
}
|
|
||||||
|
|
||||||
return colVect;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::vector<int> Grid::reverseLine(std::vector<int> line){
|
std::vector<int> Grid::reverseLine(std::vector<int> line){
|
||||||
std::vector<int> reversedLine;
|
std::vector<int> reversedLine;
|
||||||
|
|
||||||
|
@ -218,7 +210,7 @@ std::vector<int> Grid::reverseLine(std::vector<int> line){
|
||||||
return reversedLine;
|
return reversedLine;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//Return true if the grid is full. False else.
|
||||||
bool Grid::isFull(){
|
bool Grid::isFull(){
|
||||||
|
|
||||||
for(int i=0;i<m_size;i++){
|
for(int i=0;i<m_size;i++){
|
||||||
|
@ -227,10 +219,11 @@ bool Grid::isFull(){
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//Return true if the grid is full and no move can be performed. False else.
|
||||||
bool Grid::isOver(){
|
bool Grid::isOver(){
|
||||||
|
|
||||||
if(!this->isFull())
|
if(!this->isFull())
|
||||||
|
@ -242,7 +235,7 @@ bool Grid::isOver(){
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for(int i=0;i<m_size;i++){
|
for(int i=0;i<m_size;i++){
|
||||||
std::vector<int> colVect(this->getCol(i));
|
std::vector<int> colVect(this->getCol(i));
|
||||||
|
|
||||||
|
@ -256,3 +249,99 @@ bool Grid::isOver(){
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//Return true if line1 is equals to line2. False else.
|
||||||
|
bool Grid::compareLines(std::vector<int> line1, std::vector<int> line2){
|
||||||
|
|
||||||
|
for(int i=0;i<m_size;i++){
|
||||||
|
|
||||||
|
if(line1.at(i) != line2.at(i))
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
//Return the description of the grid in a string.
|
||||||
|
//In other terms, the grid in ASCII
|
||||||
|
std::string Grid::description(){
|
||||||
|
|
||||||
|
//Init stringstream description
|
||||||
|
std::stringstream description;
|
||||||
|
|
||||||
|
//Get max str len of the grid
|
||||||
|
int maxStrLen=this->maxStrLenInGrid();
|
||||||
|
|
||||||
|
//Start to write description
|
||||||
|
std::stringstream gridBorder;
|
||||||
|
for(int i=0;i<(maxStrLen+1)*4+1;i++){
|
||||||
|
gridBorder<<"-";
|
||||||
|
}
|
||||||
|
description << std::endl << gridBorder.str() << std::endl;
|
||||||
|
for(int i=0;i<m_size;i++){
|
||||||
|
for(int j=0;j<m_size;j++){
|
||||||
|
std::stringstream spaceCol;
|
||||||
|
//Get number of space needed in cell
|
||||||
|
int nbSpace=maxStrLen-std::to_string(m_grid.at(i).at(j)).size();
|
||||||
|
|
||||||
|
for(int k=0;k<nbSpace;k++){
|
||||||
|
spaceCol << " ";
|
||||||
|
}
|
||||||
|
if(m_grid.at(i).at(j) == 0)
|
||||||
|
description << "|" << spaceCol.str() << " ";
|
||||||
|
else
|
||||||
|
description << "|"<< spaceCol.str() << m_grid.at(i).at(j) ;
|
||||||
|
}
|
||||||
|
description << "|";
|
||||||
|
description << std::endl;
|
||||||
|
}
|
||||||
|
description << gridBorder.str() << std::endl << std::endl;
|
||||||
|
|
||||||
|
//Return description
|
||||||
|
return description.str();
|
||||||
|
}
|
||||||
|
|
||||||
|
//==================== Getters and Setters ====================
|
||||||
|
|
||||||
|
//Getter for m_lastMoveScore
|
||||||
|
int Grid::getLastMoveScore(){
|
||||||
|
return m_lastMoveScore;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//Change value of cell with a tuple
|
||||||
|
bool Grid::setCell(std::tuple<int, int> coord, int value){
|
||||||
|
int i=std::get<0>(coord);
|
||||||
|
int j=std::get<1>(coord);
|
||||||
|
|
||||||
|
if(i>=0 && i<m_size && j>=0 && j<m_size){
|
||||||
|
m_grid.at(i).at(j)=value;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
//Change value of cell with int
|
||||||
|
bool Grid::setCell(int i, int j, int value){
|
||||||
|
std::tuple<int, int> coord(i,j);
|
||||||
|
return this->setCell(coord, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
//Assign a vector to a column.
|
||||||
|
void Grid::setCol(int col, std::vector<int> colVect){
|
||||||
|
for(int i=0;i<m_size;i++){
|
||||||
|
m_grid.at(i).at(col)=colVect.at(i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//Retrieve a specific column.
|
||||||
|
std::vector<int> Grid::getCol(int col){
|
||||||
|
|
||||||
|
std::vector<int> colVect;
|
||||||
|
|
||||||
|
for(int i=0;i<m_size;i++){
|
||||||
|
colVect.push_back(m_grid.at(i).at(col));
|
||||||
|
}
|
||||||
|
|
||||||
|
return colVect;
|
||||||
|
}
|
||||||
|
|
|
@ -15,38 +15,46 @@
|
||||||
class Grid
|
class Grid
|
||||||
{
|
{
|
||||||
private:
|
private:
|
||||||
|
//Members
|
||||||
int m_size;
|
int m_size;
|
||||||
std::vector<std::vector<int> > m_grid;
|
std::vector<std::vector<int> > m_grid;
|
||||||
|
int m_lastMoveScore;
|
||||||
|
|
||||||
|
//Private methods
|
||||||
int maxStrLenInGrid();
|
int maxStrLenInGrid();
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
//Constructor and Destructor
|
||||||
Grid();
|
Grid();
|
||||||
~Grid();
|
~Grid();
|
||||||
|
|
||||||
std::string description();
|
//Defragment and merge methods
|
||||||
bool isEmpty(int i, int j);
|
std::vector<int> rightDefragment(std::vector<int> line);
|
||||||
std::tuple<int, int> getRandomEmptyCellCoord();
|
std::vector<int> leftDefragment(std::vector<int> line);
|
||||||
|
std::vector<int> rightMerge(std::vector<int> line);
|
||||||
|
std::vector<int> leftMerge(std::vector<int> line);
|
||||||
|
|
||||||
bool setCell(std::tuple<int, int> coord, int value);
|
//Swipe methods
|
||||||
bool setCell(int i, int j, int value);
|
bool swipeRight();
|
||||||
|
bool swipeLeft();
|
||||||
std::vector<int> swipeLine(std::vector<int> line);
|
bool swipeUp();
|
||||||
std::vector<int> defragmentLine(std::vector<int> line);
|
bool swipeDown();
|
||||||
std::vector<int> mergeLine(std::vector<int> line);
|
|
||||||
|
|
||||||
std::vector<int> getCol(int col);
|
|
||||||
|
|
||||||
|
//Helpers
|
||||||
bool isFull();
|
bool isFull();
|
||||||
bool isOver();
|
bool isOver();
|
||||||
|
bool isEmpty(int i, int j);
|
||||||
void setCol(int col, std::vector<int> colVect);
|
std::tuple<int, int> getRandomEmptyCellCoord();
|
||||||
|
bool compareLines(std::vector<int> line1, std::vector<int> line2);
|
||||||
std::vector<int> reverseLine(std::vector<int> line);
|
std::vector<int> reverseLine(std::vector<int> line);
|
||||||
|
std::string description();
|
||||||
|
|
||||||
//Moves
|
//Getters and Setters
|
||||||
void swipeRight();
|
bool setCell(std::tuple<int, int> coord, int value);
|
||||||
void swipeLeft();
|
bool setCell(int i, int j, int value);
|
||||||
void swipeUp();
|
std::vector<int> getCol(int col);
|
||||||
void swipeDown();
|
void setCol(int col, std::vector<int> colVect);
|
||||||
|
int getLastMoveScore();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -25,6 +25,6 @@ int main()
|
||||||
//Run the game
|
//Run the game
|
||||||
controller.run();
|
controller.run();
|
||||||
|
|
||||||
|
//End the application
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue