【小游戏】用C++结合EasyX制作扫雷3

开头

本次修改了格子周围8个格子的方法,原理是创建比雷盘更大一圈的二维vector,从而使每个格子周围都有8个格子,而且不会越界。不像之前特殊位置的格子做特殊判断,那样十分麻烦(受到CSDN博主 初阶牛 的文章 《c语言实现扫雷(详细讲解)》启发)

另外,去掉了鼠标指针移动到格子上时格子高亮的功能(不然比较卡顿)

下面是源代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
#include<graphics.h>
#include<iostream>
#include<vector>
#include<string>

using namespace std;
//全局变量
MOUSEMSG msg;//全局鼠标信息变量
IMAGE mine, grid, grid_motion, grid_down, flag, flag_motion , mine_other;//所有图像变量
//单个格子大小(单位:像素),雷数常量,格子布局长宽(单位:格子),窗口宽度,窗口高度
int grid_size = 40, layout_size = 10, width = 400, height = 400;

int all_num = 0, grid_x = 0, grid_y = 0, mines_num = 10;

//格子类,边长为40像素的正方形
class Grid {
public:
bool is_mine, is_click = false, is_flag = false;
int num = 0;
IMAGE mine;
int x1, x2, y1, y2;//格子的四个角顶点的位置
int i, j;
int grids[8][2] = { 0 };

void init(bool is_mine, int x, int y, int i, int j) {
//初始化
this->is_mine = is_mine;
x1 = x; x2 = x + grid_size;
y1 = y; y2 = y + grid_size;
this->i = i; this->j = j;
//绘制
change(grid);
//以此格子为中心
int grids[8][2] = { {i - 1,j - 1} ,{i - 1,j} ,{i - 1,j + 1} ,{i,j - 1} ,{i,j + 1} ,{i + 1,j - 1} ,{i + 1,j} ,{i + 1,j + 1} };
for (int a = 0; a < 8; a++) {
this->grids[a][0] = grids[a][0];
this->grids[a][1] = grids[a][1];
}
}

void change(IMAGE image) {
putimage(x1, y1, &image);
}

void show_num() {
if (num == 0)
return;
else if (num == 1)
settextcolor(BLUE);
else if (num == 2)
settextcolor(GREEN);
else if (num == 3)
settextcolor(YELLOW);
else if (num == 4)
settextcolor(RGB(255, 135, 35));
else
settextcolor(RED);
setbkmode(TRANSPARENT);
settextstyle(45, 0, _T("Consolas"));
char info = num + 48;
outtextxy(x1 + 6, y1 - 3, _T(info));
}

void find_0(vector<vector<Grid>> &layout) {
for (int a = 0; a < 8; a++) {//遍历grids(周围格子)
int i = grids[a][0], j = grids[a][1];
if (layout[i][j].num == 0 && layout[i][j].is_click == false && layout[i][j].is_mine == false) {
//模拟点击操作
layout[i][j].is_flag = false;
layout[i][j].change(grid_down);
layout[i][j].is_click = true;
all_num++;
layout[i][j].find_0(layout);
}
}
}
};

class Button {
public:
int x, y, width, height;
const char* text="NULL";

void show(const char* text, int x, int y, int width, int height, COLORREF color1 = WHITE)
{
this->x = x; this->y = y;
this->width = width; this->height = height;
this->text = text;

setlinecolor(color1);//设置框边颜色
setbkmode(TRANSPARENT);//设置字体背景透明
setfillcolor(color1);//设置填充颜色
fillroundrect(x, y, x + width, y + height, 10, 10);//画一个按钮框

settextcolor(BLACK);
settextstyle(40, 0, "黑体");
int tx = x + (width - textwidth(text)) / 2;
int ty = y + (height - textheight(text)) / 2;
outtextxy(tx, ty, text);
}

void change(COLORREF color2=RGB(237, 237, 237)) {
setlinecolor(color2);//设置框边颜色
setfillcolor(color2);//设置填充颜色
fillroundrect(x, y, x + width, y + height, 10, 10);//画一个按钮框

settextcolor(BLACK);
settextstyle(40, 0, "黑体");
int tx = x + (width - textwidth(text)) / 2;
int ty = y + (height - textheight(text)) / 2;
outtextxy(tx, ty, text);
}
};

void settings() {
int temp=0;
cout << "设置格子大小(单位:像素)" << endl;
cin >> grid_size;
cout << "设置雷盘边长(单位:格子)" << endl;
cin >> layout_size;
width = height = layout_size * grid_size;
cout << width << ' ' << height << endl;
cout << "设置雷数" << endl;
cin >> temp;
if (temp >= layout_size * layout_size)
cout << "设置的雷太多了" << endl;
else {
mines_num = temp;
cout << "OK 开始!" << endl;
return;
}
}

void start() {
//程序初始化
initgraph(width, height);
setbkcolor(BLACK);
cleardevice();
setbkcolor(WHITE);
//绘制设置按钮

Button btn1;
btn1.show("设置", 150, 100, 100, 50);
Button btn2;
btn2.show("开始", 150, 200, 100, 50);
//死循环
while (true) {
ExMessage msg = getmessage(EX_MOUSE);

if (msg.message == WM_LBUTTONDOWN) {
if (msg.x >= btn1.x && msg.x <= btn1.x + btn1.width && msg.y >= btn1.y && msg.y <= btn1.y + btn1.height) {
btn1.change();
settings();
cleardevice();
initgraph(width, height);
return;
}

else if (msg.x >= btn2.x && msg.x <= btn2.x + btn2.width && msg.y >= btn2.y && msg.y <= btn2.y + btn2.height) {
btn2.change();
cleardevice();
return;
}
}
}
}

int main() {
//程序初始化
start();
//加载图像
loadimage(&mine, "images/mine.png", grid_size, grid_size);
loadimage(&grid, "images/grid.png", grid_size, grid_size);
loadimage(&grid_down, "images/grid_down.png", grid_size, grid_size);
loadimage(&flag, "images/flag.png", grid_size, grid_size);
//布雷
vector<vector<Grid>> layout(layout_size+2, vector<Grid>(layout_size+2));//布局,用二维数组,如果是1就有雷,0则没有

srand(time(nullptr));
//初始化
for (int i = 1; i < layout_size+1; i++) {
for (int j = 1; j < layout_size+1; j++) {
layout[i][j].init(false, grid_x, grid_y, i, j);
grid_x += grid_size;
}
grid_y += grid_size;
grid_x = 0;
}
//随机布雷实现
int x = 0, y = 0, count = 0;
for (count = 0; count < mines_num;){
x = 1 + rand() % layout_size+1;
y = 1 + rand() % layout_size+1;
if (layout[x][y].is_mine == false){
layout[x][y].is_mine = true;
count++;//每次布置好一个雷之后,才会计数
}
}

for (int i = 0; i < layout_size; i++) {
for (int j = 0; j < layout_size; j++) {
//每个格子获取周围格子的雷数
for (int a = 0; a < 8; a++) {
if (layout[layout[i][j].grids[a][0]][layout[i][j].grids[a][1]].is_mine) {
layout[i][j].num++;
}
}
}
}
/*调试输出布局
for (int i = 0; i < layout_size; i++) {
for (int j = 0; j < layout_size; j++) {
cout << layout[i][j].is_mine << ' ';
}
cout << endl;
}*/
//轮询监测鼠标事件
while (true) {
//监测鼠标操作
ExMessage msg = getmessage(EX_MOUSE);

if (msg.message == WM_LBUTTONDOWN) {
for (int i = 1; i < layout_size+1; i++) {
for (int j = 1; j < layout_size+1; j++) {
//通过点击坐标判断是哪个格子
if (msg.x >= layout[i][j].x1 && msg.x <= layout[i][j].x2 && msg.y >= layout[i][j].y1 && msg.y <= layout[i][j].y2) {
if (layout[i][j].is_click == false) { //判断是否被点击过
layout[i][j].is_flag = false;
//如果这个格子是雷
if (layout[i][j].is_mine) {
layout[i][j].change(mine);
//显示提示信息
for (int x = 0; x < layout_size; x++) {
for (int y = 0; y < layout_size; y++) {
if (layout[x][y].is_mine && x != i && y != j)
layout[x][y].change(mine_other);
}
}
cout << "踩雷!即将退出";
Sleep(2500); //停顿两秒半
//退出程序
return 0;
}

else {
layout[i][j].change(grid_down);
//显示格子周围的雷数
layout[i][j].show_num();
//如果是数量是0
if (layout[i][j].num == 0)
layout[i][j].find_0(layout);
}
layout[i][j].is_click = true;
all_num++;
}
//判断未点开的格子数,从而判断输赢
if (all_num == layout_size * layout_size - mines_num) {
//显示提示信息
for (int x = 0; x < layout_size; x++) {
for (int y = 0; y < layout_size; y++) {
if (layout[x][y].is_mine && x != i && y != j)
layout[x][y].change(flag);
}
}
cout<<"获胜!即将退出";
Sleep(2500); //停顿两秒半
//退出
return 0;
}
}
}
}
}

else if (msg.message == WM_RBUTTONDOWN) {
for (int i = 1; i < layout_size+1; i++) {
for (int j = 1; j < layout_size+1; j++) {
//通过点击坐标判断是哪个格子
if (msg.x >= layout[i][j].x1 && msg.x <= layout[i][j].x2 && msg.y >= layout[i][j].y1 && msg.y <= layout[i][j].y2) {
//如果已经被标记就取消标记
if (layout[i][j].is_flag) {
layout[i][j].is_flag = false;
layout[i][j].change(grid);
}
//如果没有被标记就标记上
else if (layout[i][j].is_flag == false && layout[i][j].is_click == false) {
layout[i][j].is_flag = true;
layout[i][j].change(flag);
}
}
}
}
}
}

closegraph();
return 0;
}

结尾

项目详见GitHub:clear-sea/MineSweeper(github.com)

(欢迎给颗star)