Merge branch 'develop'

-Correct bugs
	-Manage score
	-Add some helpers
	-Optimize some code
This commit is contained in:
manzerbredes 2015-05-03 09:58:01 +02:00
commit 8488827025
8 changed files with 353 additions and 177 deletions

View file

@ -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;
}

View file

@ -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

View file

@ -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;
} }

View file

@ -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;
}

View file

@ -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

View file

@ -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;
}

View file

@ -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();
}; };

View file

@ -25,6 +25,6 @@ int main()
//Run the game //Run the game
controller.run(); controller.run();
//End the application
return 0; return 0;
} }