- Part I :Graphical part
- Part II : Core of the game (algorithm)
- Part III :MVC + database Implementation
- Last Words..
This is an example of how you may give instructions on setting up your project locally. To get a local copy up and running follow these simple example steps.
- How to install Qt
- Clone the repo
git clone https://github.com/IlyasKadi/2048-Game.git
In this part we worked mostly with ui
to design the form of the parts of te GAME.
So basically there is a simple image of the game and two buttons one to launch a new game
and the second one for checking scores
So when clicking first button it will leads you to :
This is our main game interface
we decided to keep it as much classic
and minimalist as possible one label for 2048
and two others one for best score
and the other for the realtime score
and a button to reset
in case you get bored or you don't like your progess .
The idea id to work with a 4*4 matrix to store the values of the board and corresponding each number 2,4,8..2048 to a tile (label well designed)
This is the code part that made that first look
Those are the main function used in the main game view
settile(int numberintile);
putOnnums();
setMainBorder();
setinitialpos();
//Design of tile based on each number
QLabel * NumsGame:: settile(int numberintile)
{
QString labelNum = QString::number(numberintile);
QLabel *tile = new QLabel(labelNum);
tile->setAlignment(Qt::AlignCenter);
switch (numberintile) {
case 2: {
tile->setStyleSheet("background: rgb(238,228,218);" "color: rgb(119,110,101);" "font: bold; border-radius: 10px; font: 22pt;");
break;
}
case 4: {
tile->setStyleSheet("background: rgb(237,224,200);" "color: rgb(119,110,101);" "font: bold; border-radius: 10px; font: 22pt;");
break;
}
case 8: {
tile->setStyleSheet("background: rgb(242,177,121);" "color: rgb(255,255,255);" "font: bold; border-radius: 10px; font: 22pt;");
break;
}
case 16: {
tile->setStyleSheet("background: rgb(245,150,100);" "color: rgb(255,255,255);" "font: bold; border-radius: 10px; font: 22pt;");
break;
}
case 32: {
tile->setStyleSheet("background: rgb(245,125,95);" "color: rgb(255,255,255);" "font: bold; border-radius: 10px; font: 22pt;");
break;
}
case 64: {
tile->setStyleSheet("background: rgb(245,95,60);" "color: rgb(255,255,255);" "font: bold; border-radius: 10px; font: 22pt;");
break;
}
case 128: {
tile->setStyleSheet("background: rgb(237,207,114);" "color: rgb(255,255,255);" "font: bold;" "border-radius: 10px; font: 22pt;");
break;
}
case 256: {
tile->setStyleSheet("background: rgb(237,204,97);" "color: rgb(255,255,255);" "font: bold; border-radius: 10px; font: 22pt;");
break;
}
case 512: {
tile->setStyleSheet("background: rgb(237,204,97);" "color: rgb(255,255,255);" "font: bold; border-radius: 10px; font: 22pt;");
break;
}
case 1024: {
tile->setStyleSheet("background: rgb(237,204,97);" "color: rgb(255,255,255);" "font: bold; border-radius: 10px; font: 22pt;");
break;
}
case 2048: {
QGraphicsDropShadowEffect *dse = new QGraphicsDropShadowEffect();
dse->setColor(Qt::yellow);
dse->setBlurRadius(50);
dse->setOffset(-1);
tile->setGraphicsEffect(dse);
tile->setStyleSheet("background: rgb(237,204,97);" "color: rgb(255,255,255); font: bold;" "border-radius: 10px; font: 22pt;");
break;
}
default: {
tile = new QLabel();
tile->setStyleSheet("background: rgb(205,192,180);" "border-radius: 10px;" "color: rgb(119,110,101);");
break;
}
}
return tile;
}
//Corresponding each num with its tile designed
void NumsGame::putOnnums()
{
//clear all
for(int i=0; i<4 ;i++)
{
for(int j=0; j<4 ;j++)
{
ui->gridboard->addWidget(settile(0),i,j);
}
}
//Put on tiles
for(int i=0; i<4 ;i++)
{
for(int j=0; j<4 ;j++)
{
if(numsMatrix[i][j]!=0)
ui->gridboard->addWidget(settile(numsMatrix[i][j]),i,j);
}
}
}
///Setting up base view
void NumsGame::setMainBorder()
{
//initializing board (matrix 4*4)
numsMatrix.resize(4);
for (int i = 0; i < 4; i++)
numsMatrix[i].resize(4);
for (int i = 0; i < 4; i++)
for (int j = 0; j < 4;j++)
numsMatrix[i][j]=0;
ui->gameoverlabel->hide();
ui->winner->hide();
ui->Tryagain->hide();
ui->newGame->hide();
ui->Quit->hide();
ui->lastScore->hide();
ui->youreScore->hide();
ui->nicknamelabel->hide();
ui->Nickname->hide();
ui->submit->hide();
for (int i = 0; i < 4; ++i) {
for (int j = 0; j < 4; ++j) {
QLabel *label = new QLabel();
label->setStyleSheet("background: rgb(205,192,180);" "border-radius: 10px;" "color: rgb(119,110,101);" );
label->setAlignment(Qt::AlignCenter);
ui->gridboard->addWidget(label,i,j);
}
}
setinitialpos();
}
//Forming a random position
std::pair<int, int> NumsGame::formrandpos()
{
int randi = rand() % 4;
int randj = rand() % 4;
return std::make_pair(randj, randi);
}
//Setting up two first tiles to start with
void NumsGame::setinitialpos()
{
auto [rndi,rndj]=formrandpos();
auto [rndi_,rndj_]=formrandpos();
numsMatrix[rndi][rndj]=2;
numsMatrix[rndi_][rndj_]=2;
ui->gridboard->addWidget(settile(2),rndi,rndi);
ui->gridboard->addWidget(settile(2),rndi_,rndj_);
}
void NumsGame::start()
{
this->setFixedSize(this->geometry().width(),this->geometry().height());
getbestscore();
setMainBorder();
}
Otherwise if you clicked second button
it will leads you to :
This is the High Scores interface
two simples labels one for Nickname
and the other for High Scores
and a list view
where we going to put our model
.
And for the main meal there is a lot to talk about .., we were able to implement our own algorithm (inspired by the original one) in the game (and by the way Oussama used to play the GAME a lot) that's why we decided to keep it as first choice instead of working on another game.
So first things first :
We made an easy movement logic
that is divided into three phases
(for each movement in a certain direction) :
Remove Extra Spaces
Perform The Sum
Remove Extra Spaces
Here is a Simple schema
that explain how it really works within an exemple of moving up :
Same thing for other movement you just need to play a bit with indexes
A first case when spaces needed to be removed first
A first case when spaces needed to be removed after the sum
And for the coding part :
move up |
move down |
void NumsGame::moveUp()
{
oldscore=score;
diffscore=0;
oladboard=numsMatrix;
//this is a space remover phase
for (int j = 0; j < 4; j++)
{
for (int i = 1; i < 4; i++)
{
if (numsMatrix[i][j] != 0)
{
for(int k=0;k<i;k++)
{
if (numsMatrix[k][j] == 0)
{
numsMatrix[k][j]=numsMatrix[i][j];
numsMatrix[i][j]=0;
}
}
}
}
}
//this is the sum phase (and so on concerning other dorections)
for (int j = 0; j < 4; j++)
{
for (int i = 1; i < 4; i++)
{
if (numsMatrix[i][j] != 0)
{
if(numsMatrix[i-1][j] == numsMatrix[i][j] )
{
numsMatrix[i-1][j]=numsMatrix[i][j]*2;
score+=numsMatrix[i-1][j];
numsMatrix[i][j]=0;
}
}
}
}
//this is another space remover phase (and the same for other dorections)
for (int j = 0; j < 4; j++)
{
for (int i = 1; i < 4; i++)
{
if (numsMatrix[i][j] != 0)
{
for(int k=0;k<i;k++)
{
if (numsMatrix[k][j] == 0)
{
numsMatrix[k][j]=numsMatrix[i][j];
numsMatrix[i][j]=0;
}
}
}
}
}
putOnnums();
move_or_die();
diffscore=score-oldscore;
if(diffscore >0 ){ ScoreAddedSayHi(diffscore);};
}
|
void NumsGame::moveDown()
{
oldscore=score;
diffscore=0;
oladboard=numsMatrix;
for(int j=0;j<4;j++)
{
for(int i=2;i>=0;i--)
{
if(numsMatrix[i][j]!=0)
{
for(int k=3;k>i;k--)
{
if(numsMatrix[k][j]==0)
{
numsMatrix[k][j]=numsMatrix[i][j];
numsMatrix[i][j]=0;
}
}
}
}
}
for(int j=0;j<4;j++)
{
for(int i=2;i>=0;i--)
{
if(numsMatrix[i][j]!=0)
{
if(numsMatrix[i+1][j]==numsMatrix[i][j])
{
numsMatrix[i+1][j]=numsMatrix[i][j]*2;
score+=numsMatrix[i+1][j];
numsMatrix[i][j]=0;
}
}
}
}
for(int j=0;j<4;j++)
{
for(int i=2;i>=0;i--)
{
if(numsMatrix[i][j]!=0)
{
for(int k=3;k>i;k--)
{
if(numsMatrix[k][j]==0)
{
numsMatrix[k][j]=numsMatrix[i][j];
numsMatrix[i][j]=0;
}
}
}
}
}
putOnnums();
move_or_die();
diffscore=score-oldscore;
if(diffscore >0 ){ ScoreAddedSayHi(diffscore);};
}
|
move Right |
move left |
void NumsGame::moveRight()
{
oldscore=score;
diffscore=0;
oladboard=numsMatrix;
for(int i=0;i<4;i++)
{
for(int j=2;j>=0;j--)
{
if(numsMatrix[i][j]!=0)
{
for(int k=3;k>j;k--)
{
if(numsMatrix[i][k]==0)
{
numsMatrix[i][k]=numsMatrix[i][j];
numsMatrix[i][j]=0;
}
}
}
}
}
for(int i=0;i<4;i++)
{
for(int j=2;j>=0;j--)
{
if(numsMatrix[i][j]!=0)
{
if(numsMatrix[i][j+1]==numsMatrix[i][j] )
{
numsMatrix[i][j+1]=numsMatrix[i][j]*2;
score+=numsMatrix[i][j+1];
numsMatrix[i][j]=0;
}
}
}
}
for(int i=0;i<4;i++)
{
for(int j=2;j>=0;j--)
{
if(numsMatrix[i][j]!=0)
{
for(int k=3;k>j;k--)
{
if(numsMatrix[i][k]==0)
{
numsMatrix[i][k]=numsMatrix[i][j];
numsMatrix[i][j]=0;
}
}
}
}
}
putOnnums();
move_or_die();
diffscore=score-oldscore;
if(diffscore >0 ){ ScoreAddedSayHi(diffscore);};
}
|
void NumsGame::moveLeft()
{
oldscore=score;
diffscore=0;
oladboard=numsMatrix;
for(int i=0;i<4;i++)
{
for(int j=1;j<4;j++)
{
if(numsMatrix[i][j]!=0)
{
for(int k=0;k<j;k++)
{
if(numsMatrix[i][k]==0)
{
numsMatrix[i][k]=numsMatrix[i][j];
numsMatrix[i][j]=0;
}
}
}
}
}
for(int i=0;i<4;i++)
{
for(int j=1;j<4;j++)
{
if(numsMatrix[i][j]!=0)
{
if(numsMatrix[i][j-1]==numsMatrix[i][j] )
{
numsMatrix[i][j-1]=numsMatrix[i][j]*2;
score+=numsMatrix[i][j-1];
numsMatrix[i][j]=0;
}
}
}
}
for(int i=0;i<4;i++)
{
for(int j=1;j<4;j++)
{
if(numsMatrix[i][j]==0)
{
for(int k=0;k<j;k++)
{
if(numsMatrix[i][k]==0)
{
numsMatrix[i][k]=numsMatrix[i][j];
numsMatrix[i][j]=0;
}
}
}
}
}
putOnnums();
move_or_die();
diffscore=score-oldscore;
if(diffscore >0 ){ ScoreAddedSayHi(diffscore);};
}
|
And this is the link
between movement
and keyboard keys
:
dontmove
is a variable that allows player to move as long as he can move You didn't finish yet (neither a winner or loser)
void NumsGame::keyPressEvent(QKeyEvent *event)
{
switch (event->key())
{
if(dontmove==0)
{
case Qt::Key_Up:
{
moveUp();
ui->resetbutton->setEnabled(1);
break;
}
case Qt::Key_Left:
{
moveLeft();
ui->resetbutton->setEnabled(1);
break;
}
case Qt::Key_Right:
{
moveRight();
ui->resetbutton->setEnabled(1);
break;
}
case Qt::Key_Down:
{
moveDown();
ui->resetbutton->setEnabled(1);
break;
}
}
}
}
This is a special part of move_or_die
function that decide if a new tile is going to be created after an attempt to move to a certain direction so that player can move ( before winning or losing the game ) if yes it creates a new tile in a random place.
So basically it compare between two matrices the old one before pressing a key (in an direction) and the new one after pressing THE SAME KEY so if there is no change in THE MATRIX you can't have a new tile while you keep pressing trying to go in that same direction otherwise you can easaly win..-_-..
This is an example where you can't move down anymore
//condition that check if next move is possible :
if( oladboard!=numsMatrix) //movement possible
{
//updating score:
ui->Score_N->setText(QString::number(score));
if(ui->BEST_SCORE_N->text().toInt()<=score)
{
bscore=score;
ui->BEST_SCORE_N->setText(QString::number(score));
}
//making a rand free position
std:: pair<int, int> randpos =formrandpos();
do {
randpos =formrandpos();
} while (numsMatrix[randpos.first][randpos.second] != 0);
//to add a new tile in it:
numsMatrix[randpos.first][randpos.second]=2 ;
ui->gridboard->addWidget(settile(2),randpos.first,randpos.second);
}
Winners see the gain ------ losers see the pain.
Yeah.. next phase is win and lose logic :
So easy you win when you reach 2048.
And you lose if there is no movement possible to do :
- All tiles are full and there is no possible sum to perform.
Code :
void NumsGame:: move_or_die()
{
int c=0;
//No_Tile_Left loop check
for (int i = 0; i < 4; i++)
{
for (int j = 0; j < 4;j++)
{
if( numsMatrix[i][j]==0)
{
c++;
}
}
}
//No_Sum_possible_ loop check by cols
for (int i = 0; i < 4; i++)
{
for (int j = 0; j < 3;j++)
{
if( numsMatrix[i][j]== numsMatrix[i][j+1])
{
c++;
}
}
}
//No_Sum_possible_ loop check by rows
for (int j = 0; j < 4; j++)
{
for (int i = 0; i < 3;i++)
{
if( numsMatrix[i][j]== numsMatrix[i+1][j])
{
c++;
}
}
}
//Win loop check
for (int i = 0; i < 4; i++)
{
for (int j = 0; j < 4;j++)
{
if( numsMatrix[i][j]==2048)
{
winner();
dontmove=1;
}
}
}
//condition that check if next move is possible :
if( oladboard!=numsMatrix) //movement possible
{
//updating score:
ui->Score_N->setText(QString::number(score));
if(ui->BEST_SCORE_N->text().toInt()<=score)
{
bscore=score;
ui->BEST_SCORE_N->setText(QString::number(score));
}
//making a rand free position
std:: pair<int, int> randpos =formrandpos();
do {
randpos =formrandpos();
} while (numsMatrix[randpos.first][randpos.second] != 0);
//to add a new tile in it:
numsMatrix[randpos.first][randpos.second]=2 ;
ui->gridboard->addWidget(settile(2),randpos.first,randpos.second);
}
else if(oladboard==numsMatrix && c==0) // movement impossible you're dead +_+
{
gameOver();
dontmove=1;
}
// otherwise do nothing (won't add a tile)
}
void NumsGame::ScoreAddedSayHi(int i)
{
ui->scoreadded->setText("+"+QString::number(i));
ui->scoreadded->show();
QTimer::singleShot(500, ui->scoreadded, &QLabel::hide);
}
THIS IS HOW IT LOOKS WHEN YOU WIN OR LOSE
This is where you finish your game and get your score, or maybe you are still in the game and watch your score going up and up ..
Those are two variables and two lists that concern this part :
-
int score
-
int bscore
-
oldscore
; -
diffscore
;
-
QStringList scoreslist
; -
QVector <int> scoreslistnum
;
For the score it increases every time you add two or more tiles :
for(int i=0;i<4;i++)
{
for(int j=1;j<4;j++)
{
if(numsMatrix[i][j]!=0)
{
if(numsMatrix[i][j-1]==numsMatrix[i][j] )
{
numsMatrix[i][j-1]=numsMatrix[i][j]*2;
score+=numsMatrix[i][j-1]; //score added
numsMatrix[i][j]=0;
}
}
}
}
And for the best score it gets its value from a database of scores :
void NumsGame::getbestscore()
{
db =QSqlDatabase::addDatabase("QSQLITE");
db.setDatabaseName("/Users/pc/Desktop/scores_.sqlite");
db.open();
QSqlQuery scores("SELECT * from score ",db);
while(scores.next())
scoreslist.append(scores.value(1).toString()+"");
for(QString e: scoreslist)
scoreslistnum.push_back(e.toInt());
ui->BEST_SCORE_N->setText(QString::number(scoreslistnum[scoreslistnum.size()-1]));
}
And it changes everytime you hit better score than the best (not best anymore) if movement is possible :
//condition that check if next move is possible :
if( oladboard!=numsMatrix) //movement possible
{
//updating score:
ui->Score_N->setText(QString::number(score));
if(ui->BEST_SCORE_N->text().toInt()<=score)
{
bscore=score;
ui->BEST_SCORE_N->setText(QString::number(score));
}
.
.
.
}
A beutifull little detail was added :
Each time you add up two or 2^n tile a small label shows that amount of 2^n added quickly and disapear :
This is how it looks like :
void NumsGame::ScoreAddedSayHi(int i)
{
ui->scoreadded->setText("+"+QString::number(i));
ui->scoreadded->show();
QTimer::singleShot(500, ui->scoreadded, &QLabel::hide);
}
We put it inside movement function so that everytime a score is added ( diffscore > 0) it says hi and go
oldscore=score;
diffscore=0;
.
.
.
putOnnums();
move_or_die();
diffscore=score-oldscore;
if(diffscore >0 ){ ScoreAddedSayHi(diffscore);};
This is the part where MVC
intervene we decided to keep it away from the main game because [ the table view ] is the only model that will carry the game but absolutely won't keep its beauty and classical character (we don't think that somone would do when it means a lot to him and after he was so addicted with it and then he grows up and get the chance to make his own), but instead we decide to use a list view that shows the score (stored in a database
) .
This is how it looks like :
Code :
Setting up database :
void NumsGame::setdatabase(QString nickname, int score)
{
db =QSqlDatabase::addDatabase("QSQLITE");
db.setDatabaseName("/Users/pc/Desktop/scores_.sqlite");
db.open();
QSqlQuery query(db);
QString create {"CREATE TABLE IF NOT EXISTS score (name VARCHAR(80), score int)"};
if(!query.exec(create))
{
QMessageBox::critical(this,"info","could not create table");
}
QString insert {"INSERT INTO score values ('%1','%2')"};
if(!query.exec(insert.arg(nickname).arg(score)))
{
QMessageBox::critical(this,"info","insert not create table");
}
}
Fill up db on_submit_clicked :
void NumsGame::on_submit_clicked()
{
setdatabase(ui->Nickname->text(),score);
reset();
ui->Nickname->setText("");
}
Model :
QStandardItemModel *Scorelistmodel= nullptr;
highscores.cpp
HighScores::HighScores(QWidget *parent) :
QDialog(parent),
ui(new Ui::HighScores)
{
ui->setupUi(this);
Scorelistmodel = new QStandardItemModel;
loadscores();
}
HighScores::~HighScores()
{
delete ui;
}
void HighScores::loadscores()
{
NumsGame newgame;
newgame.db =QSqlDatabase::addDatabase("QSQLITE");
newgame.db.setDatabaseName("/Users/pc/Desktop/scores_.sqlite");
newgame.db.open();
QSqlQuery scores("SELECT DISTINCT * from score order by score desc",newgame.db);
while(scores.next())
newgame.scoreslisttoload.append(" "+scores.value(0).toString() + " " + scores.value(1).toString()+"");
for(auto e :newgame.scoreslisttoload)
{
QString path{""};
QIcon icon(path);
ui->scoreview->setModel(Scorelistmodel);
Scorelistmodel->appendRow(new QStandardItem(QIcon(icon),e));
}
}
As shown above, the query to fill up the listview makes sure to put on the scores in a DESC order and DISTINCT values to avoid redundancy.
And last but not least we just want to THANK YOU .. a lot we don't think that words could describe how much glad we are to have such brilliant and helpful professor, two years wasn't enough to get bored of you and we don't think that another 50 years will, but is was enough to make us more productive and filled with your beutifull, big, powerful energy that you never doubt giving it to us.. plus the huge amount of informations that you get sad if you don't give it all to us.... so THANK YOU so much for being there always for us and THANK YOU for everything that you give us starting with a simple smile or even a Hi with your eyes.
Our Team : AIT EL KADI Ilyas - AZIZ Oussama
Project Link : The 2048 Game
Encadré par : Mr.BELCAID-Anass