一、演示效果
二、核心代码
#include "GameModel.h"
#include <time.h>
#include <stdlib.h>
GameModel::GameModel(){
}
void GameModel::startGame(GameType type){
gameType = type;
gameMapVec.clear();
for(int i=0;i<BOARD_GRAD_SIZE;i++){
std::vector<int> lineBoard;
for(int j=0;j<BOARD_GRAD_SIZE;j++){
lineBoard.push_back(0);
}
gameMapVec.push_back(lineBoard);
}
if(gameType == AI){
scoreMapVec.clear();
for(int i=0;i<BOARD_GRAD_SIZE;i++){
std::vector<int> lineScores;
for(int j=0;j<BOARD_GRAD_SIZE;j++){
lineScores.push_back(0);
}
scoreMapVec.push_back(lineScores);
}
}
playerFlag = true;
}
void GameModel::actionByPerson(int row, int col){
updateGameMap(row,col);
}
void GameModel::updateGameMap(int row, int col){
if(playerFlag){
gameMapVec[row][col] = 1;
}else{
gameMapVec[row][col] = -1;
}
playerFlag = !playerFlag;
}
bool GameModel::isWin(int row,int col){
for(int i=0;i<5;i++){
if(row>0 && row<BOARD_GRAD_SIZE &&
col-i>0 && col-i+4<BOARD_GRAD_SIZE &&
gameMapVec[row][col-i] == gameMapVec[row][col-i+1] &&
gameMapVec[row][col-i] == gameMapVec[row][col-i+2] &&
gameMapVec[row][col-i] == gameMapVec[row][col-i+3] &&
gameMapVec[row][col-i] == gameMapVec[row][col-i+4]){
return true;
}
if(row-i>0 && row-i+4<BOARD_GRAD_SIZE &&
col>0 && col<BOARD_GRAD_SIZE &&
gameMapVec[row-i][col] == gameMapVec[row-i+1][col] &&
gameMapVec[row-i][col] == gameMapVec[row-i+2][col] &&
gameMapVec[row-i][col] == gameMapVec[row-i+3][col] &&
gameMapVec[row-i][col] == gameMapVec[row-i+4][col]){
return true;
}
if(
row-i>0 && row-i+4<BOARD_GRAD_SIZE &&
col+i-4>0 && col+i<BOARD_GRAD_SIZE &&
gameMapVec[row-i][col+i] == gameMapVec[row-i+1][col+i-1] &&
gameMapVec[row-i][col+i] == gameMapVec[row-i+2][col+i-2] &&
gameMapVec[row-i][col+i] == gameMapVec[row-i+3][col+i-3] &&
gameMapVec[row-i][col+i] == gameMapVec[row-i+4][col+i-4]){
return true;
}
if(row-i>0 && row-i+4<BOARD_GRAD_SIZE &&
col-i>0 && col-i+4<BOARD_GRAD_SIZE &&
gameMapVec[row-i][col-i] == gameMapVec[row-i+1][col-i+1] &&
gameMapVec[row-i][col-i] == gameMapVec[row-i+2][col-i+2] &&
gameMapVec[row-i][col-i] == gameMapVec[row-i+3][col-i+3] &&
gameMapVec[row-i][col-i] == gameMapVec[row-i+4][col-i+4]){
return true;
}
}
return false;
}
void GameModel::calculateScore(){
int personNum = 0;
int botNum = 0;
int emptyNum = 0;
scoreMapVec.clear();
for(int i=0;i<BOARD_GRAD_SIZE;i++){
std::vector<int> lineScores;
for(int j=0;j<BOARD_GRAD_SIZE;j++){
lineScores.push_back(0);
}
scoreMapVec.push_back(lineScores);
}
for(int row=0;row<BOARD_GRAD_SIZE;row++){
for(int col=0;col<BOARD_GRAD_SIZE;col++){
if(row>0 && col>0 && gameMapVec[row][col]==0){
for(int y=-1;y<=1;y++){
for(int x=-1;x<=1;x++){
personNum = 0;
botNum = 0;
emptyNum = 0;
if(!(y==0 && x==0)){
for(int i=1;i<=4;i++){
if(row+i*y>0 && row+i*y<BOARD_GRAD_SIZE &&
col+i*x>0 && col+i*x<BOARD_GRAD_SIZE &&
gameMapVec[row+i*y][col+i*x]==1){
personNum++;
}else if(row+i*y>0 && row+i*y<BOARD_GRAD_SIZE &&
col+i*x>0 && col+i*x<BOARD_GRAD_SIZE &&
gameMapVec[row+i*y][col+i*x]==0){
emptyNum++;
break;
}else{
break;
}
}
for(int i=1;i<=4;i++){
if(row-i*y>0 && row-i*y<BOARD_GRAD_SIZE &&
col-i*x>0 && col-i*x<BOARD_GRAD_SIZE &&
gameMapVec[row-i*y][col-i*x]==1){
personNum++;
}else if(row-i*y>0 && row-i*y<BOARD_GRAD_SIZE &&
col-i*x>0 && col-i*x<BOARD_GRAD_SIZE &&
gameMapVec[row-i*y][col-i*x]==0){
emptyNum++;
break;
}else{
break;
}
}
if(personNum == 1){
scoreMapVec[row][col]+=10;
}else if(personNum == 2){
if(emptyNum == 1)
scoreMapVec[row][col]+=30;
else if(emptyNum == 2)
scoreMapVec[row][col]+=40;
}else if(personNum == 3){
if(emptyNum == 1)
scoreMapVec[row][col]+=60;
else if(emptyNum == 2)
scoreMapVec[row][col]+=110;
}else if(personNum == 4){
scoreMapVec[row][col]+=10100;
}
emptyNum = 0;
for(int i=1;i<=4;i++){
if(row+i*y>0 && row+i*y<BOARD_GRAD_SIZE &&
col+i*x>0 && col+i*x<BOARD_GRAD_SIZE &&
gameMapVec[row+i*y][col+i*x]==-1){
botNum++;
}else if(row+i*y>0 && row+i*y<BOARD_GRAD_SIZE &&
col+i*x>0 && col+i*x<BOARD_GRAD_SIZE &&
gameMapVec[row+i*y][col+i*x]==0){
emptyNum++;
break;
}else{
break;
}
}
for(int i=1;i<=4;i++){
if(row-i*y>0 && row-i*y<BOARD_GRAD_SIZE &&
col-i*x>0 && col-i*x<BOARD_GRAD_SIZE &&
gameMapVec[row-i*y][col-i*x]==-1){
botNum++;
}else if(row-i*y>0 && row-i*y<BOARD_GRAD_SIZE &&
col-i*x>0 && col-i*x<BOARD_GRAD_SIZE &&
gameMapVec[row-i*y][col-i*x]==0){
emptyNum++;
break;
}else{
break;
}
}
if(botNum == 0){
scoreMapVec[row][col]+=5;
}else if(botNum == 1){
scoreMapVec[row][col]+=10;
}else if(botNum == 2){
if(emptyNum == 1)
scoreMapVec[row][col]+=25;
else if(emptyNum == 2)
scoreMapVec[row][col]+=50;
}else if(botNum == 3){
if(emptyNum == 1)
scoreMapVec[row][col]+=55;
else if(emptyNum == 2)
scoreMapVec[row][col]+=100;
}else if(botNum >= 4){
scoreMapVec[row][col]+=20000;
}
}
}
}
}
}
}
}
void GameModel::actionByAI(int &clickRow,int &clickCol){
calculateScore();
int maxScore = 0;
std::vector<std::pair<int,int>> maxPoints;
for(int row = 1;row<BOARD_GRAD_SIZE;row++){
for(int col = 1;col<BOARD_GRAD_SIZE;col++){
if(gameMapVec[row][col] == 0){
if(scoreMapVec[row][col]>maxScore){
maxPoints.clear();
maxScore = scoreMapVec[row][col];
maxPoints.push_back(std::make_pair(row,col));
}else if(scoreMapVec[row][col] == maxScore){
maxPoints.push_back(std::make_pair(row,col));
}
}
}
}
srand((unsigned)time(0));
int index = rand()%maxPoints.size();
std::pair<int,int> pointPair = maxPoints.at(index);
clickRow = pointPair.first;
clickCol = pointPair.second;
updateGameMap(clickRow,clickCol);
}
#include "mainwindow.h"
#include "ui_mainwindow.h"
#include <QPainter>
#include <QMouseEvent>
#include <math.h>
#include <QMessageBox>
#include <QTimer>
#include <QDebug>
MainWindow::MainWindow(QWidget *parent)
: QMainWindow(parent)
, ui(new Ui::MainWindow)
{
ui->setupUi(this);
this->setFixedSize(
MARGIN*2+BLOCK_SIZE*BOARD_GRAD_SIZE,
MARGIN*2+BLOCK_SIZE*BOARD_GRAD_SIZE);
initGame();
}
MainWindow::~MainWindow()
{
delete ui;
}
void MainWindow::paintEvent(QPaintEvent* event){
QPainter painter(this);
painter.setRenderHint(QPainter::Antialiasing);
for(int i=0;i<BOARD_GRAD_SIZE+1;i++){
painter.drawLine(MARGIN+BLOCK_SIZE*i,MARGIN,
MARGIN+BLOCK_SIZE*i,this->height()-MARGIN);
painter.drawLine(MARGIN,MARGIN+BLOCK_SIZE*i,
this->width()-MARGIN,MARGIN+BLOCK_SIZE*i);
}
QBrush brush;
brush.setStyle(Qt::SolidPattern);
if(clickPosRow>0 && clickPosRow<BOARD_GRAD_SIZE &&
clickPosCol>0 && clickPosCol<BOARD_GRAD_SIZE &&
game->gameMapVec[clickPosRow][clickPosCol]==0){
if(game->playerFlag){
brush.setColor(Qt::black);
}else{
brush.setColor(Qt::white);
}
painter.setBrush(brush);
painter.drawRect(MARGIN+BLOCK_SIZE*clickPosCol-MARK_SIZE/2,MARGIN+BLOCK_SIZE*clickPosRow-MARK_SIZE/2,MARK_SIZE,MARK_SIZE);
}
for(int i=0;i<BOARD_GRAD_SIZE;i++){
for(int j=0;j<BOARD_GRAD_SIZE;j++){
if(game->gameMapVec[i][j]==1){
brush.setColor(Qt::black);
painter.setBrush(brush);
painter.drawEllipse(MARGIN+BLOCK_SIZE*j-CHESS_RADIUS,MARGIN+BLOCK_SIZE*i-CHESS_RADIUS,CHESS_RADIUS*2,CHESS_RADIUS*2);
}else if(game->gameMapVec[i][j]==-1){
brush.setColor(Qt::white);
painter.setBrush(brush);
painter.drawEllipse(MARGIN+BLOCK_SIZE*j-CHESS_RADIUS,MARGIN+BLOCK_SIZE*i-CHESS_RADIUS,CHESS_RADIUS*2,CHESS_RADIUS*2);
}
}
}
if(clickPosCol>0 && clickPosCol<BOARD_GRAD_SIZE &&
clickPosRow>0 && clickPosRow<BOARD_GRAD_SIZE &&
(game->gameMapVec[clickPosRow][clickPosCol]==1||game->gameMapVec[clickPosRow][clickPosCol]==-1)){
if(game->isWin(clickPosRow,clickPosCol) && game->gameStatus == PLAYING){
game->gameStatus = WIN;
QString str;
str = game->gameMapVec[clickPosRow][clickPosCol]==1?u8"黑棋":u8"白棋";
QMessageBox::StandardButton btnValue = QMessageBox::information(this,u8"五子棋嬴家",str+u8"勝利");
if(btnValue == QMessageBox::Ok){
game->startGame(game_type);
game->gameStatus = PLAYING;
}
}
}
}
void MainWindow::initGame(){
game = new GameModel();
QMessageBox::StandardButton select = QMessageBox::question(this,u8"選擇遊戲模式",u8"YES[玩家VS電腦] NO[玩家VS玩家]");
if(select == QMessageBox::Yes){
game_type = AI;
}else {
game_type = MAN;
}
game->gameStatus = PLAYING;
game->startGame(game_type);
update();
}
void MainWindow::mouseMoveEvent(QMouseEvent* event){
int x = event->x();
int y = event->y();
if(x>=MARGIN+BLOCK_SIZE/2&&
x<this->width()-MARGIN-BLOCK_SIZE/2&&
y>=MARGIN+BLOCK_SIZE/2&&
y<this->height()-MARGIN-BLOCK_SIZE/2){
int col = (x-MARGIN)/BLOCK_SIZE;
int row = (y-MARGIN)/BLOCK_SIZE;
int leftTopPosX = MARGIN+BLOCK_SIZE*col;
int leftTopPosY = MARGIN+BLOCK_SIZE*row;
clickPosRow = -1;
clickPosCol = -1;
int len = 0;
selectPos = false;
len = sqrt((x-leftTopPosX)*(x-leftTopPosX)+(y-leftTopPosY)*(y-leftTopPosY));
if(len<POS_OFFSET){
clickPosRow = row;
clickPosCol = col;
if(game->gameMapVec[clickPosRow][clickPosCol]==0){
selectPos = true;
}
}
len = sqrt((x-leftTopPosX-BLOCK_SIZE)*(x-leftTopPosX-BLOCK_SIZE)+(y-leftTopPosY)*(y-leftTopPosY));
if(len<POS_OFFSET){
clickPosRow = row;
clickPosCol = col+1;
if(game->gameMapVec[clickPosRow][clickPosCol]==0){
selectPos = true;
}
}
len = sqrt((x-leftTopPosX)*(x-leftTopPosX)+(y-leftTopPosY-BLOCK_SIZE)*(y-leftTopPosY-BLOCK_SIZE));
if(len<POS_OFFSET){
clickPosRow = row+1;
clickPosCol = col;
if(game->gameMapVec[clickPosRow][clickPosCol]==0){
selectPos = true;
}
}
len = sqrt((x-leftTopPosX-BLOCK_SIZE)*(x-leftTopPosX-BLOCK_SIZE)+(y-leftTopPosY-BLOCK_SIZE)*(y-leftTopPosY-BLOCK_SIZE));
if(len<POS_OFFSET){
clickPosRow = row+1;
clickPosCol = col+1;
if(game->gameMapVec[clickPosRow][clickPosCol]==0){
selectPos = true;
}
}
}
update();
}
void MainWindow::mouseReleaseEvent(QMouseEvent* event){
if(selectPos == false){
return;
}else{
selectPos = false;
}
chessOneByPerson();
if(game_type == AI){
QTimer::singleShot(AI_THINK_TIME,this,SLOT(chessOneByAI()));
}
}
void MainWindow::chessOneByPerson(){
if(clickPosRow!=-1 && clickPosCol!=-1 && game->gameMapVec[clickPosRow][clickPosCol]==0){
game->actionByPerson(clickPosRow,clickPosCol);
update();
}
}
void MainWindow::chessOneByAI(){
game->actionByAI(clickPosRow,clickPosCol);
update();
}
三、下载链接
https://download.csdn.net/download/u013083044/89656663