Easyx实现扫雷游戏

本文实例为大家分享了Easyx实现扫雷游戏的具体代码,供大家参考,具体内容如下

代码:

#include<stdio.h>
#include<stdlib.h>
#include<time.h>
#include<easyx.h>
 
#include<mmsystem.h>
#pragma comment(lib, "winmm.lib")
 
#define ROW 10    //定义行列的常量
#define COL    10 
#define MineNum 10    //雷的数量
#define ImgSize    40    //图片的尺寸
 
//定义图片资源
IMAGE imgs[12];
void loadResource()
{
    for (int i = 0; i < 12; i++)
    {
        char imgPath[50] = { 0 };
        sprintf_s(imgPath, "./images/%d.jpg", i);
        loadimage(&imgs[i], imgPath, ImgSize, ImgSize);
    }
}
 
bool isfirst = true;    //是不是第一次进来
 
//函数声明
void show(int map[][COL]);
void init(int map[][COL]);
void draw(int map[][COL]);
void mouseMsg(ExMessage* msg, int map[][COL]);
void boomBlank(int map[][COL], int row, int col);
int judge(int map[][COL], int row, int col);
int main()
{
    //创建窗口
    initgraph(400, 400/*,EW_SHOWCONSOLE*/);
    //播放开始音乐
    mciSendString("open ./images/start.mp3 alias bgm", NULL, 0, NULL);
    mciSendString("play bgm", NULL, 0, NULL);
 
    //扫雷地图
    int map[ROW][COL] = {0};
    init(map);
 
    //游戏主循环
    while (true)
    {
        //处理消息
        ExMessage msg;
        while (peekmessage(&msg, EM_MOUSE))
        {
            switch (msg.message)
            {
            case WM_LBUTTONDOWN:    //鼠标左键和右键点击
            case WM_RBUTTONDOWN:
                mouseMsg(&msg, map);
                int ret = judge(map,msg.y/ImgSize, msg.x / ImgSize);        //点击之后判断
                if (ret == -1)
                {
 
                    draw(map);
                    int select = MessageBox(GetHWnd(), "你这么牛,怎么输了呢?敢再来一把吗?", "low B!", MB_OKCANCEL);
                    if (select == IDOK)    //再来一把
                    {
                        //重新初始化
                        init(map);
                    }
                    else  //退出
                    {
                        exit(0);
                    }
 
                }
                else if(ret == 1)
                {
 
                }
                system("cls");
                printf("judege:%d\n", ret);
                show(map);
                break;
            }
        }
        draw(map);        
    }
 
    //show(map);
    getchar();
    return 0;
}
 
void show(int map[][COL])
{
    for (int i = 0; i < ROW; i++)
    {
        for (int k = 0; k < COL; k++)
        {
            printf("%2d ", map[i][k]);
        }
        printf("\n");
    }
}
//初始化数据
void init(int map[][COL])
{
    loadResource();
    //设置随机数种子
    srand((unsigned)time(NULL));
 
    //把map全部初始化为0
    memset(map, 0, sizeof(int) * ROW * COL);
 
    //随机设置十个雷 用-1表示
    for (int i = 0; i < MineNum; )
    {
        //数组的有效下标 [0,9]
        int r = rand() % ROW;
        int c = rand() % COL;
        if (map[r][c] == 0)
        {
            map[r][c] = -1;
            //只有执行了这里的代码,才成功设置了雷 -1
            i++;
        }
    }
 
    //把以雷为中心的九宫格数据都+1,雷除外
    for (int i = 0; i < ROW; i++)
    {
        for (int k = 0; k < COL; k++)
        {
            //找到雷,并遍历雷所在的九宫格
            if (map[i][k] == -1)
            {
                for (int r = i-1; r <= i+1; r++)
                {
                    for (int c = k-1; c <= k+1; c++)
                    {
                        //对周围的数据加1,会有一个bug
                        if ((r >= 0 && r < ROW && c >= 0 && c < COL) && map[r][c] != -1)
                        {
                            ++map[r][c];
                        }
                    }
                }
            }
        }
    }
 
    //加密格子
    for (int i = 0; i < ROW; i++)
    {
        for (int k = 0; k < COL; k++)
        {
            map[i][k] += 20;
        }
    }
}
//绘制
void draw(int map[][COL])
{
    //贴图,根据map里面的数据,贴对应的图片
    for (int i = 0; i < ROW; i++)
    {
        for (int k = 0; k < COL; k++)
        {
            if (map[i][k]>=0 && map[i][k]<=8)    //[0,8]
            {
                int index = map[i][k];    //0 1 2 3 4 5 6 7 8
                putimage(k * ImgSize, i * ImgSize, &imgs[index]);    //
            }
            else if (map[i][k] == -1)
            {
                putimage(k * ImgSize, i * ImgSize, &imgs[9]);
            }
            else if (map[i][k] >= 19 && map[i][k] <= 28)
            {
                putimage(k * ImgSize, i * ImgSize, &imgs[10]);
            }
            else if(map[i][k] >= 39)    //-1 + 20 +20
            {
                putimage(k * ImgSize, i * ImgSize, &imgs[11]);
            }
        }
    }
}
//鼠标操作数据
void mouseMsg(ExMessage* msg,int map[][COL])
{
    //先根据鼠标点击的坐标求出对应的数组的下标
    int r = msg->y / ImgSize;
    int c = msg->x / ImgSize;
    //左键打开格子
    if (msg->message == WM_LBUTTONDOWN)
    {
        //什么时候能够打开,没有打开的时候就打开
        if (map[r][c]>=19 && map[r][c]<=28)
        {
            //这个函数只能播放wav格式
            PlaySound("./images/click.wav", NULL, SND_ASYNC | SND_FILENAME);
            map[r][c] -= 20;
            boomBlank(map, r, c);    //检测一下是不是空白格子,是,炸开
            isfirst = true;
        }    
    }
    //右键标记格子
    else if (msg->message == WM_RBUTTONDOWN)
    {
        PlaySound("./images/rightClick.wav", NULL, SND_ASYNC | SND_FILENAME);
        //是否能够标记:如果没有打开就能标记
        if (map[r][c] >= 19 && map[r][c] <= 28)
        {
            map[r][c] += 20;
        }
        else if(map[r][c]>=39)
        {
            map[r][c] -= 20;
        }
    }
}
//点击空白格子,连环爆开周围的所有空白格子还有数字  row col 是当前点击的格子
void boomBlank(int map[][COL],int row,int col)
{    
    //判断row col位置是不是空白格子
    if (map[row][col] == 0)
    {
        for (int r = row-1; r <= row+1; r++)
        {
            for (int c = col-1; c <= col+1; c++)
            {
                if ((r>=0&&r<ROW&&c>=0&&c<COL)            //没越界
                    && map[r][c]>=19 && map[r][c]<=28)    //没有打开
                {
                    //每一次调用都会播放一下
                    if (isfirst)
                    {
                        PlaySound("./images/search.wav", NULL, SND_ASYNC | SND_FILENAME);
                        isfirst = false;
                    }
 
                    map[r][c] -= 20;
                    boomBlank(map, r, c);
                }
            }
        }
    }
    return;
}
//游戏结束条件 输了返回-1  没结束返回0 赢了返回 1
int judge(int map[][COL],int row ,int col)
{
    //点到了雷,结束    输了
    if (map[row][col] == -1 || map[row][col] == 19)
    {
        return -1;
    }
 
    //点完了格子,结束 赢了 点开了100 - 10 = 90 个格子
    int cnt = 0;
    for (int i = 0; i < ROW; i++)
    {
        for (int k = 0; k < COL; k++)
        {
            //统计打开的格子的数量
            if (map[i][k] >= 0 && map[i][k] <= 8)
            {
                ++cnt;
            }
        }
    }
    if (ROW*COL - MineNum == cnt)
    {
        return 1;
    }
 
    return 0;
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • C++基于EasyX实现简单扫雷游戏

    本文实例为大家分享了C++ EasyX实现简单扫雷游戏的具体代码,供大家参考,具体内容如下 [实现代码] #include <cmath> #include <time.h> #include <easyx.h> #include <conio.h> using namespace std; #define Size 500 //定义窗口大小 #define SquareSize 50 //定义格子大小 #define BackGroundColor LIG

  • 利用c++和easyx图形库做一个低配版扫雷游戏

    游戏界面 由于这个游戏是我抱着玩一玩的心态做出来的,所以没有过多的去设计界面,也没有去找游戏的资源(图片.游戏音效等).仅使用了不同颜色的方块来表示游戏中方块的状态和种类.(绿色为初始状态(未翻转的状态),黄色为翻转后的背景颜色,蓝色表示已插旗的方块,红色代表地雷) 图1 游戏主菜单界面 图二 模式一的游戏界面(20*20 40个雷) 图三 模式二的游戏界面(10*10 20个雷) 图四 游戏成功界面 图五 游戏失败界面 2.全部代码 #include<graphics.h> #include

  • Easyx实现扫雷游戏

    本文实例为大家分享了Easyx实现扫雷游戏的具体代码,供大家参考,具体内容如下 代码: #include<stdio.h> #include<stdlib.h> #include<time.h> #include<easyx.h>   #include<mmsystem.h> #pragma comment(lib, "winmm.lib")   #define ROW 10    //定义行列的常量 #define COL 

  • C++代码实现扫雷游戏

    前言 提示:本文是基于easyX图形库实现的,还有部分功能可以添加,仅适合新手参考. 提示:以下是本篇文章正文内容,下面案例可供参考 一.扫雷游戏模式 在确定大小的矩形雷区中随机布置一定数量的地雷,玩家需要尽快找出雷区中的所有不是地雷的方块,而不许踩到地雷. 游戏的基本操作包括左键单击和右键单击.其中左键用于打开安全的格子,推进游戏进度:右键用于标记地雷,以辅助判断. 左键单击:在判断出不是雷的方块上按下左键,可以打开该方块.如果方块上出现数字,则该数字表示其周围3×3区域中的地雷数(一般为8个

  • 基于C语言扫雷游戏的设计与实现

    目录 1 引言 2 相关工作 3 本文方法 4 结果与分析 5 总结 整体代码 1 引言 伴随着信息技术的快速发展,近年来,人们的生活已经离不开计算机.生活娱乐几乎都是在计算机上进行的.其中的扫雷游戏就是之一.扫雷游戏是微软公司在1992年在windows系统上发布的一款益智类小游戏,直到今天这款小游戏依然存在,可见此款游戏的成功.本文将用Visual Studio 2019作为开发工具来模拟实现扫雷游戏.经过大一第一学期的学习,我们对C语言的理论及其相关知识有了一定的认识和了解.本文可以把我们

  • php实现的简易扫雷游戏实例

    本文实例讲述了php实现的简易扫雷游戏.分享给大家供大家参考.具体如下: <?php $init = $_POST["init"];//game restart $clickvalue = $_POST["clickvalue"];//minesweeping $checkflag = 0;//Victory or defeat $click_count = 0;//clicks count if($init == null && $click

  • 详解从零开始---用C#制作扫雷游戏

    学C#的原因其实挺简单的,因为一直对游戏挺感兴趣,查了下比较流行的游戏引擎Unity的主要开发语言是C#,所以就决定从C#入手,学学面向对象的编程方法. 以前基本都做的是嵌入式开发,做嵌入式久了,基本上只用C语言,C语言面向过程的特性在嵌入式编程这种资源极度受限的情况确实十分有利,但这种方式在面对大型软件的开发的时候就很难胜任了.编程的模式其实是一种思维习惯,习惯久了以后,想改变确实是一个艰难的过程··· 说起C#,其实在大学的时候学过一个学期,说来惭愧那时候倒也没把它当一门面向对象的语言(其实

  • C++实现简单的扫雷游戏(控制台版)

    C++新手的代码,请各位多包涵. 用C++写的一个简单的控制台版扫雷游戏.玩家通过输入方块的坐标来翻开方块. 只是一个雏形,能够让玩家执行翻开方块的操作并且判断输赢,还未添加标记方块.游戏菜单.记录游戏时间.重新开一局等等的功能. 玩家输入坐标的方式来翻开方块只适用于小型的"雷区",若"雷区"大了,用坐标会变得很不方便. 代码片段扫雷V1.1 #include<stdio.h> #include<Windows.h> #define YELL

  • JavaScript版经典游戏之扫雷游戏完整示例【附demo源码下载】

    本文实例讲述了JavaScript扫雷游戏.分享给大家供大家参考,具体如下: 翻出年初写的游戏贴上来,扫雷相信大家都玩过,先上图: 源码: <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> <html xmlns="http://www.w3.or

  • 基于C语言实现的扫雷游戏代码

    本文详细讲述了基于C语言实现的扫雷游戏代码,代码中备有比较详细的注释,便于读者阅读和理解.希望对学习游戏开发的朋友能有一点借鉴价值. 完整的实例代码如下: /* 模拟扫雷游戏 */ #include <graphics.h> #include <math.h> #include <stdio.h> #include <dos.h> #include <stdlib.h> #include <conio.h> #include <

  • java swing实现的扫雷游戏及改进版完整示例

    本文实例讲述了java swing实现的扫雷游戏及改进版.分享给大家供大家参考,具体如下: 版本1: package awtDemo; import java.awt.event.ActionEvent; import java.awt.event.ActionListener; import java.awt.event.MouseEvent; import java.awt.event.MouseListener; import javax.swing.JButton; import jav

随机推荐