diff --git a/src/Controllers/ConsoleController/ConsoleController.cpp b/src/Controllers/ConsoleController/ConsoleController.cpp index d91e807..1bc9b84 100644 --- a/src/Controllers/ConsoleController/ConsoleController.cpp +++ b/src/Controllers/ConsoleController/ConsoleController.cpp @@ -2,14 +2,22 @@ #include #include "../../Helpers/Keyboard.hpp" + +//==================== Constructor and Destructor ==================== + +//Constructor ConsoleController::ConsoleController() { } +//Destructor ConsoleController::~ConsoleController() { } +//==================== Helpers ==================== + +//Run the game. void ConsoleController::run() { @@ -22,6 +30,9 @@ void ConsoleController::run() //Pop a random number on the grid m_game.popRandomNumber(); + //First cout stats + this->coutStats(); + //First cout grid m_game.coutGrid(); @@ -33,20 +44,27 @@ void ConsoleController::run() keyPress=this->waitArrowKeyPress(); //Apply move - m_game.swipe(keyPress); + bool moveDone=m_game.swipe(keyPress); - //Pop a random number on the grid - m_game.popRandomNumber(); + //Cout stats + this->coutStats(); //Cout grid m_game.coutGrid(); } + + //Last cout stats + this->coutStats(); + + //Last cout grid m_game.coutGrid(); } + +//Wait for keypress and return the keyPress. kbdh::Direction ConsoleController::waitArrowKeyPress() { //Initialise keyPress @@ -59,7 +77,8 @@ kbdh::Direction ConsoleController::waitArrowKeyPress() keyPress=kbdh::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; } @@ -68,7 +87,8 @@ kbdh::Direction ConsoleController::waitArrowKeyPress() keyPress=kbdh::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; } @@ -77,17 +97,18 @@ kbdh::Direction ConsoleController::waitArrowKeyPress() keyPress=kbdh::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; } if (sf::Keyboard::isKeyPressed(sf::Keyboard::Down)) { - // la touche "flèche gauche" est enfoncée : on bouge le personnage keyPress=kbdh::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; } @@ -95,3 +116,11 @@ kbdh::Direction ConsoleController::waitArrowKeyPress() 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; +} diff --git a/src/Controllers/ConsoleController/ConsoleController.hpp b/src/Controllers/ConsoleController/ConsoleController.hpp index 8d73f79..f3e9952 100644 --- a/src/Controllers/ConsoleController/ConsoleController.hpp +++ b/src/Controllers/ConsoleController/ConsoleController.hpp @@ -18,9 +18,13 @@ class ConsoleController Game m_game; kbdh::Direction waitArrowKeyPress(); public: + //Constructor and Destructor ConsoleController(); ~ConsoleController(); + + //Helpers void run(); + void coutStats(); }; #endif diff --git a/src/Helpers/Keyboard.hpp b/src/Helpers/Keyboard.hpp index 9d2dfa3..c24c1ba 100644 --- a/src/Helpers/Keyboard.hpp +++ b/src/Helpers/Keyboard.hpp @@ -10,6 +10,7 @@ namespace kbdh { //Key arrow enum Direction { Up, Down, Left, Right }; + //Key arrow typedef typedef enum Direction Direction; } diff --git a/src/Model/Game.cpp b/src/Model/Game.cpp index 6039bc9..284d82b 100644 --- a/src/Model/Game.cpp +++ b/src/Model/Game.cpp @@ -1,49 +1,85 @@ #include "Game.hpp" +//==================== Constructor and Destructor ==================== -Game::Game() : m_grid(){ +//Constructor +Game::Game() : m_grid(), m_score(0), m_nbMove(0){ } +//Destructor Game::~Game(){ } +//==================== Helpers ==================== - +//Swipe action bool Game::swipe(kbdh::Direction direction){ + bool moveDone; + switch(direction){ case kbdh::Left: - m_grid.swipeLeft(); + moveDone=m_grid.swipeLeft(); break; case kbdh::Right: - m_grid.swipeRight(); + moveDone=m_grid.swipeRight(); break; case kbdh::Up: - m_grid.swipeUp(); + moveDone=m_grid.swipeUp(); break; case kbdh::Down: - m_grid.swipeDown(); + moveDone=m_grid.swipeDown(); break; } - return true; + if(moveDone){ + m_score+=m_grid.getLastMoveScore(); + m_nbMove++; + this->popRandomNumber(); + } + + return moveDone; } +//Cout the grid void Game::coutGrid(){ std::cout << m_grid.description(); } +//Return true if the game is lost. False else. bool Game::isOver(){ return m_grid.isOver(); } +//Pop a random number on the grid void Game::popRandomNumber(){ std::tuple 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); } +//==================== Getters and Setter ==================== + +//Retrieve the Score +int Game::getScore(){ + return m_score; +} + +//Retrieve the number of moves +int Game::getNbMove(){ + return m_nbMove; +} diff --git a/src/Model/Game.hpp b/src/Model/Game.hpp index ab19340..0b2ee4d 100644 --- a/src/Model/Game.hpp +++ b/src/Model/Game.hpp @@ -16,16 +16,25 @@ class Game { private: + //Members Grid m_grid; + int m_score; + int m_nbMove; public: + //Constructor and Destructor Game(); ~Game(); - + + //Helpers bool swipe(kbdh::Direction direction); void coutGrid(); void popRandomNumber(); bool isOver(); + + //Getters and Setters + int getScore(); + int getNbMove(); }; #endif diff --git a/src/Model/Grid.cpp b/src/Model/Grid.cpp index 3a8a075..cc0fe60 100644 --- a/src/Model/Grid.cpp +++ b/src/Model/Grid.cpp @@ -1,5 +1,7 @@ #include "Grid.hpp" +//==================== Constructor and Destructor ==================== + //Constructor 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); } } - } //Destructor @@ -17,41 +18,140 @@ Grid::~Grid(){ } -std::string Grid::description(){ +//==================== Merge and defragment methods ==================== - //Init stringstream description - std::stringstream description; +//Defragment the line to the right +std::vector Grid::rightDefragment(std::vector line){ + for(int j=0; jmaxStrLenInGrid(); - - //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 Grid::leftDefragment(std::vector line){ + + std::vector reversedLine= this->reverseLine(line); + + return this->reverseLine(this->rightDefragment(reversedLine)); +} + +//Merge line to the right +std::vector Grid::rightMerge(std::vector 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 Grid::leftMerge(std::vector 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 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 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 colVect=this->getCol(i); + + std::vector 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 colVect=this->getCol(i); + + std::vector 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 max=0; for(int i=0;i Grid::getRandomEmptyCellCoord(){ //Init list of candidate @@ -97,117 +199,7 @@ std::tuple Grid::getRandomEmptyCellCoord(){ } - - -//Change value of cell -bool Grid::setCell(std::tuple coord, int value){ - int i=std::get<0>(coord); - int j=std::get<1>(coord); - - if(i>=0 && i=0 && j coord(i,j); - return this->setCell(coord, value); -} - - -std::vector Grid::defragmentLine(std::vector line){ - for(int j=0; j Grid::mergeLine(std::vector 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 Grid::swipeLine(std::vector 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; iswipeLine(m_grid.at(i)); - } -} - -//Swipe to right -void Grid::swipeLeft(){ - for(int i=0; ireverseLine(this->swipeLine(this->reverseLine(m_grid.at(i)))); - } -} - - -void Grid::swipeUp(){ - for(int i=0; i colVect=this->getCol(i); - this->setCol(i,this->reverseLine(this->swipeLine(this->reverseLine(colVect)))); - } -} -void Grid::swipeDown(){ - for(int i=0; i colVect=this->getCol(i); - this->setCol(i,this->swipeLine(colVect)); - } -} - -void Grid::setCol(int col, std::vector colVect){ - for(int i=0;i Grid::getCol(int col){ - - std::vector colVect; - - for(int i=0;i Grid::reverseLine(std::vector line){ std::vector reversedLine; @@ -218,7 +210,7 @@ std::vector Grid::reverseLine(std::vector line){ return reversedLine; } - +//Return true if the grid is full. False else. bool Grid::isFull(){ for(int i=0;iisFull()) @@ -242,7 +235,7 @@ bool Grid::isOver(){ return false; } } - + for(int i=0;i 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 line1, std::vector line2){ + + for(int i=0;imaxStrLenInGrid(); + + //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 coord, int value){ + int i=std::get<0>(coord); + int j=std::get<1>(coord); + + if(i>=0 && i=0 && j coord(i,j); + return this->setCell(coord, value); +} + +//Assign a vector to a column. +void Grid::setCol(int col, std::vector colVect){ + for(int i=0;i Grid::getCol(int col){ + + std::vector colVect; + + for(int i=0;i > m_grid; + int m_lastMoveScore; + //Private methods int maxStrLenInGrid(); + public: + //Constructor and Destructor Grid(); ~Grid(); - std::string description(); - bool isEmpty(int i, int j); - std::tuple getRandomEmptyCellCoord(); + //Defragment and merge methods + std::vector rightDefragment(std::vector line); + std::vector leftDefragment(std::vector line); + std::vector rightMerge(std::vector line); + std::vector leftMerge(std::vector line); - bool setCell(std::tuple coord, int value); - bool setCell(int i, int j, int value); - - std::vector swipeLine(std::vector line); - std::vector defragmentLine(std::vector line); - std::vector mergeLine(std::vector line); - - std::vector getCol(int col); + //Swipe methods + bool swipeRight(); + bool swipeLeft(); + bool swipeUp(); + bool swipeDown(); + //Helpers bool isFull(); bool isOver(); - - void setCol(int col, std::vector colVect); + bool isEmpty(int i, int j); + std::tuple getRandomEmptyCellCoord(); + bool compareLines(std::vector line1, std::vector line2); std::vector reverseLine(std::vector line); + std::string description(); - //Moves - void swipeRight(); - void swipeLeft(); - void swipeUp(); - void swipeDown(); + //Getters and Setters + bool setCell(std::tuple coord, int value); + bool setCell(int i, int j, int value); + std::vector getCol(int col); + void setCol(int col, std::vector colVect); + int getLastMoveScore(); }; diff --git a/src/main.cpp b/src/main.cpp index 49c7a20..48c1766 100644 --- a/src/main.cpp +++ b/src/main.cpp @@ -25,6 +25,6 @@ int main() //Run the game controller.run(); - + //End the application return 0; }