c++使用Easyx圖形庫實現飛機大戰

公共的頭文件        common.h

#pragma once
#include <graphics.h>
#include <iostream>
#include <string>
#include <map>
#include <list>
#include <thread>
#include <vector>
#include <ctime>
#include <mmsystem.h>
#pragma comment(lib,"winmm.lib")
using namespace std;

管理資源 res.h — 單例設計模式

#pragma once
#include "common.h"
class Res 
{
private:
	Res();    //構造函數私有化
public:       //提供公有接口
	static int WIDTH(string name);    //通過圖片的鍵,獲取資源寬度
	static int HEIGHT(string name);
	static Res* GetInstance();        //獲取資源(類)的對象---唯一的對象
	static void DrawIMG(int x, int y, string name);//畫圖片的位置,通過鍵去找畫哪張圖片
//畫角色---位置,通過姓名的方式去找  不同類型的角色,用preIndex去做標識
	static void DrawRole(int x, int y, string name, int preIndex);
//播放音樂---windows多線程 DWORD類型,WINAPI修飾
	static DWORD WINAPI PlayMusic(LPVOID lparame);
	~Res();
public:
	static map<string, IMAGE*> img;		//圖片資源---圖片存在map中
	static map<string, string> music;	//音樂資源
};

res.cpp

#include "res.h"
map<string, IMAGE*> Res::img;   //圖片資源    靜態數據成員在類外初始化,類名限定
map<string, string> Res::music;	//音樂資源
Res::Res()                      //構造函數為數據成員初始化---路徑下處理
{
//背景
    string background = "./res/background.jpg";
//角色---4張---背景圖+掩碼圖
    string roleImg[4] = { "./res/planeNormal_1.jpg","./res/planeNormal_2.jpg",
    "./res/planeExplode_1.jpg","./res/planeExplode_2.jpg" };
//子彈
    string ballImg[2] = { "./res/bullet1.jpg","./res/bullet2.jpg" };
//敵機
    string enemyImg[4] = { "./res/enemy_1.jpg","./res/enemy_2.jpg","./res/enemyPlane1.jpg","./res/enemyPlane2.jpg" };
    //string --->IMAGE* 本來就是指針,不需要取地址
    img["背景"] = new IMAGE;
    img["角色"] = new IMAGE[4];
    img["子彈"] = new IMAGE[2];
    img["敵機"] = new IMAGE[4];
    loadimage(img["背景"], background.c_str());    //加載圖片    路徑 項目屬性多字節 
    for (int i = 0; i < 4; i++)
    {
/*假設img["角色"]為p,則p=new IMAGE [4];則img["角色"]+i  等效: p+i*/
        loadimage(img["角色"] + i, roleImg[i].data());    //用.data或.cst()轉換為字符串
        loadimage(img["敵機"] + i, enemyImg[i].data());   
    }
    for (int i = 0; i < 2; i++) 
    {
        loadimage(img["子彈"] + i, ballImg[i].c_str());
    }
 
}
//獲取圖片的寬度---碰撞的時候需要---返回對象指針,對象指針調用(img類型)數據成員,有一個成員函數
int Res::WIDTH(string name)
{
//獲取對象,獲取什麼樣的屬性,(img類型)數據成員有一個getwidth()成員函數---是庫中的成員函數
    return GetInstance()->img[name]->getwidth(); 
}
//獲取圖片的高度
int Res::HEIGHT(string name)
{
    return GetInstance()->img[name]->getheight();
}
Res* Res::GetInstance()
{
    static Res* res = new Res;
    return res;
}
//隻有一張圖片的貼圖: 背景圖貼圖
void Res::DrawIMG(int x, int y, string name)
{
    putimage(x, y, GetInstance()->img[name]);    //貼圖 在x,y位置貼對象裡面的圖片
}
void Res::DrawRole(int x, int y, string name, int preIndex)
{
//多張圖片貼圖---透明貼圖---去背景
    putimage(x, y, GetInstance()->img[name] + preIndex, NOTSRCERASE);//貼第幾張---幀數
    putimage(x, y, GetInstance()->img[name] + preIndex+1, SRCINVERT);
}
DWORD __stdcall Res::PlayMusic(LPVOID lparame)
{
    int key = (int)lparame;    //線程處理函數的參數---強轉為int
    switch (key)               //不同的音樂,類型不一樣
    {
    case 1:
        mciSendString("close ./res/f_gun.mp3", 0, 0, 0);    //播放前先關閉
        mciSendString("open ./res/f_gun.mp3", 0, 0, 0);     //先打開,後播放
        mciSendString("play ./res/f_gun.mp3", 0, 0, 0);
        break;
    case 2:
        mciSendString("close ./res/5.mp3", 0, 0, 0);
        mciSendString("open ./res/5.mp3", 0, 0, 0);
        mciSendString("play ./res/5.mp3", 0, 0, 0);
        break;
    case 3:
        mciSendString("close ./res/10.mp3", 0, 0, 0);
        mciSendString("open ./res/10.mp3", 0, 0, 0);
        mciSendString("play ./res/10.mp3", 0, 0, 0);
        break;
    }
    return 0;
}
Res::~Res()
{
    delete img["背景"];
    delete[] img["角色"];
    delete[] img["敵機"];
    delete[] img["子彈"];
}

打飛機遊戲.cpp        主函數部分

#include "control.h"
#include "graph.h"
#include "role.h"
#include "enemy.h"
int main() 
{
	srand((unsigned int)time(NULL));    //隨機函數種子---位置不同
	Graph* pMap = new Graph;
	Role* pRole = new Role;
	Enemy* pEnemy = new Enemy;
	Control* pControl = new Control(pMap, pRole, pEnemy);
	BeginBatchDraw();         //雙緩沖繪圖
	while (1) 
	{
		cleardevice();        //清屏
		pControl->DrawMap();  //畫地圖
		pControl->DrawRole(); //畫角色
		pControl->DrawEnemy();//打敵機前畫敵機
		pControl->PlayEemey();
 
		FlushBatchDraw();     //顯示 執行未完成的繪制任務
	}
	EndBatchDraw();
	return 0;
}

point.h        無論是飛機移動還是子彈移動,本質都是坐標的改變 – – – 處理點的變化

#pragma once
#include "common.h"
class Point 
{
public:
	enum Dir {left,right,down,up}; //枚舉點移動的方向,不同的移動方向點的改變是不一樣的
	Point(int x = 0, int y = 0);
	Point(const Point& point);     //拷貝構造---點與點之間的賦值
	int& getX();                   //外部接口,獲取點的坐標
	int& getY();
	//移動點
	void move(int speed, Dir dir); //移動速度 方向---決定坐標如何變化
 
protected:
	int x;
	int y;
};
/* 敵機從上往下移動   角色可以上下左右移動
   s形敵機可以增加x方向增量,增加y方向增量 */

point.cpp

#include "point.h"
Point::Point(int x, int y):x(x),y(y)
{
}
Point::Point(const Point& point)
{
    this->x = point.x;
    this->y = point.y;
}
int& Point::getX()
{
    // TODO: 在此處插入 return 語句
    return x;
}
int& Point::getY()
{
    // TODO: 在此處插入 return 語句
    return y;
}
void Point::move(int speed, Dir dir)    //根據方向移動
{
    switch (dir) 
    {
    case Point::left:
        this->x -= speed;
        break;
    case Point::right:
        this->x += speed;
        break;
    case Point::up:
        this->y -= speed;
        break;
    case Point::down:
        this->y += speed;
        break;
    }
}

element.h        敵機和飛機方向不同,其他基本相同 – – – 抽象元素類

#pragma once
#include "common.h"
#include "point.h"
//所有的敵機和角色 都是由這個類進行派生的
class Element 
{
public:
	virtual ~Element();    //子類對象初始化父類指針
	Element();
	Element(int x, int y, string name, bool live, int hp = 0);
	int& GetX();
	int& GetY();
	bool& GetLive();
	int& GetHp();
	int GetWidth();        //獲取寬度,高度
	int GetHeight();
	void DrawElement(int pre);    //畫元素---畫第幾張
	void MoveElement(int speed, Point::Dir dir); //移動元素
 
protected:
	Point point;		//元素在窗口上的位置
	string name;		//元素的名字
	bool live;			//存在的標記---敵機/子彈會消失
	int hp;				//血量
};

element.cpp

#include "element.h"
#include "res.h"
int Element::GetWidth()
{
    return Res::GetInstance()->WIDTH(name);
//  return Res::GetInstance()->img[name]->getwidth();   在類IMAGE中為公有屬性
}
int Element::GetHeight()
{
    return Res::GetInstance()->HEIGHT(name);
}
//在資源文件中封裝瞭移動過程,調用資源中的函數即可---畫角色
void Element::DrawElement(int pre)
{
    Res::GetInstance()->DrawRole(point.getX(), point.getY(), name, pre);
}
//點的移動
void Element::MoveElement(int speed, Point::Dir dir) 
{
    point.move(speed, dir);
}
Element::~Element()
{
}
Element::Element()
{
 
}
//類的組合必須采用初始化參數列表
Element::Element(int x, int y, string name, bool live, int hp):point(x,y),name(name)
{
    this->live = live;
    this->hp = hp;
}
int& Element::GetX()
{
    // TODO: 在此處插入 return 語句
    return point.getX();    //用point存放這個點,獲取這個點應返回point裡面的x坐標
}
int& Element::GetY()
{
    // TODO: 在此處插入 return 語句
    return point.getY();
}
bool& Element::GetLive()
{
    // TODO: 在此處插入 return 語句
    return live;
}
int& Element::GetHp()
{
    // TODO: 在此處插入 return 語句
    return hp;
}

role.h – – – 角色移動、角色發射子彈

#pragma once
#include "common.h"
class Element;
class Role 
{
public:
	Role();
	~Role();
	void DrawPlane(int preIndex);    //第幾幀
	void MovePlane(int speed);       //速度
	void DrawBullet();               //畫子彈
	void MoveBullet(int speed);      //移動子彈---移動速度
	Element*& GetPlane();            //外部做按鍵操作,需要訪問飛機和子彈
	list<Element*>& GetBullet();  
 
protected:
	Element* plane;			//角色---用元素實例化一個角色---角色也是元素之一
	list<Element*> bullet;	//子彈---一個飛機有多個子彈(包含多個元素的對象)子彈也是元素
};

role.cpp

#include "role.h"
#include "res.h"
#include "element.h"
#include "Timer.hpp"
Role::Role()            //new一個元素類即可---把飛機放窗口正中間
{
	plane = new Element(
		Res::GetInstance()->WIDTH("背景") / 2 - Res::GetInstance()->WIDTH("角色") / 2  //x
		, Res::GetInstance()->HEIGHT("背景") - Res::GetInstance()->HEIGHT("角色"),	  //y
		"角色",	//name
		true,	//live
		100);   //hp
}
Role::~Role()
{
 
}
void Role::DrawPlane(int preIndex)    //畫飛機
{
	plane->DrawElement(preIndex);
}
void Role::MovePlane(int speed)       //移動飛機---結合按鍵控制   異步處理的按鍵操作
{
	if (GetAsyncKeyState(VK_UP) && plane->GetY() >= 0) //往上走Y不能超過上邊界
	{
		plane->MoveElement(speed, Point::up);          //改變飛機的點---移動元素
	}
                                                       //往下走<=(背景高度-角色高度)
	if (GetAsyncKeyState(VK_DOWN) &&                   
		plane->GetY()<=Res::GetInstance()->HEIGHT("背景")-Res::GetInstance()->HEIGHT("角色"))
	{
		plane->MoveElement(speed, Point::down);
	}
                                                       //往右走<=(背景寬度-角色寬度)
	if (GetAsyncKeyState(VK_RIGHT) && 
		plane->GetX() <= Res::GetInstance()->WIDTH("背景") - Res::GetInstance()->WIDTH("角色"))
	{
		plane->MoveElement(speed, Point::right);
	}
                                                        //往左走X不能小於左邊界
	if (GetAsyncKeyState(VK_LEFT) && plane->GetX() >= 0)
	{
		plane->MoveElement(speed, Point::left);
	}
	//子彈產生---按空格發射子彈---用定時器控制速度---100毫秒產生1顆子彈
	if (GetAsyncKeyState(VK_SPACE)&&MyTimer::Timer(100,0))    
	{
//添加音樂  調用Windows中的創建線程函數---函數指針 傳入線程處理函數---播放第一個音樂
		HANDLE threadID = CreateThread(NULL, 0, Res::PlayMusic, (int*)1, 0, 0);
		bullet.push_back(new Element(plane->GetX() + 45, plane->GetY() - 10, "子彈", 1));    //尾插法 按一下空格new一個子彈    子彈的坐標在飛機坐標的正上方的中間
		CloseHandle(threadID);    //通過返回值關閉線程
	}
	MoveBullet(1);    //移動子彈
	DrawBullet();     //畫子彈 
}
void Role::DrawBullet()    //子彈存在容器中,每顆子彈都要畫出來
{
	for (auto v : bullet)  //迭代器遍歷
	{
		if (v->GetLive())      //判斷子彈能否畫出來
		{
			v->DrawElement(0); //序號0,子彈隻有2張
		}
	}
}
void Role::MoveBullet(int speed)    //每顆子彈都要移動---從下往上走
{
	for (auto v : bullet)      
	{
		v->GetY() -= speed;
	}
}
Element*& Role::GetPlane()
{
	// TODO: 在此處插入 return 語句
	return plane;
}
list<Element*>& Role::GetBullet()
{
	// TODO: 在此處插入 return 語句
	return bullet;
}
//每產生一個子彈就播放音樂,返回值為HANDLE類型

control.h         控制整個遊戲的運行 – – – 中驅

#pragma once
class Graph;
class Role;
class Enemy;
class Control 
{
public:
	Control(Graph* pMap = nullptr, Role* pRole = nullptr,Enemy* pEnemy=nullptr);
	~Control();
	void DrawMap();
	void DrawRole();
	void DrawEnemy();    //畫敵機
	void PlayEemey();    //打敵機
 
protected:
	//所有組成部分
	Role* pRole;    //角色
	Graph* pMap;    //地圖
	Enemy* pEnemy;  //敵機
};

control.cpp – – – 封裝實現細節 – – – 主函數中調用控制類對象即可

#include "control.h"
#include "role.h"
#include "graph.h"    //地圖
#include "timer.hpp"
#include "enemy.h"
#include "element.h"
#include "res.h"
 
Control::Control(Graph* pMap, Role* pRole, Enemy* pEnemy )
{
	this->pMap = pMap;
	this->pRole = pRole;
	this->pEnemy = pEnemy;
}
Control::~Control()
{
 
}
void Control::DrawMap()
{
	pMap->DrawMap();
}
 
void Control::DrawRole()
{
	pRole->DrawPlane(0);    //第0幀
	pRole->MovePlane(1);    //速度
}
//每一秒產生一個敵機(產生不能過於頻繁)---產生一個敵機就把它放到容器中
void Control::DrawEnemy()
{
	if (MyTimer::Timer(1000, 1)) 
	{
		pEnemy->GetEnemy().push_back(pEnemy->createEnemy());
	}
	if (MyTimer::Timer(10, 2))   //十毫秒移動一個飛機   2號定時器
	{
		pEnemy->MoveEnemy(1);    //速度
	}
	pEnemy->DrawEnemy(0);        //隻畫1種敵機
}
 
void Control::PlayEemey()        //矩形與矩形的判斷
{
	//1.碰撞處理: 碰到子彈,把子彈的live置為0---隻處理標記
	for (auto bullet : pRole->GetBullet()) //獲取子彈的信息
	{
		if (bullet->GetLive() == 0)        //如果子彈標記==0繼續往下找
			continue;
		if (bullet->GetY() < 0)            //如果超出窗口邊界,把標記置為false
			bullet->GetLive() = false;
		for (auto enemy : pEnemy->GetEnemy()) //與子彈的碰撞處理 大前提---敵機存在
		{
			if(enemy->GetLive()&&
				bullet->GetX()>enemy->GetX()&&
				bullet->GetX()<enemy->GetX()+Res::WIDTH("敵機")&&
				bullet->GetY() > enemy->GetY() &&
				bullet->GetY() < enemy->GetY() + Res::HEIGHT("敵機")) 
			{
				enemy->GetHp()--;            //相交,血量減少
				if (enemy->GetHp() <= 0)
				{
					enemy->GetLive() = false; //把敵機標記置為false---不做消失處理
				}
				bullet->GetLive() = false;    //碰到敵機後子彈要消失   
			}
			//敵機出瞭窗口邊界要消失
			if (enemy->GetY() >= Res::HEIGHT("背景")) 
			{
				enemy->GetLive() = false;
			}
		}
	}
	//2.通過標記去刪除相應的數據--->內存管理    遍歷敵機
	for (auto iterE = pEnemy->GetEnemy().begin(); iterE != pEnemy->GetEnemy().end();) 
	{
		if ((*iterE)->GetLive() == false)     //當前敵機標記為false--->刪除敵機
		{
			iterE = pEnemy->GetEnemy().erase(iterE);
		}
		else 
		{
			iterE++;        //++不要寫在for循環中
		}
	}
//遍歷子彈---子彈刪除
	for (auto iterB = pRole->GetBullet().begin(); iterB != pRole->GetBullet().end();)
	{
		if ((*iterB)->GetLive() == false)
		{
			iterB = pRole->GetBullet().erase(iterB);
		}
		else
		{
			iterB++;
		}
	}
}

graph.h – – – 地圖(窗口類)

#pragma once
class Graph 
{
public:
	Graph();
	~Graph();
	void DrawMap();
};

graph.cpp

#include "graph.h"
#include "res.h"
Graph::Graph()
{
	initgraph(Res::GetInstance()->img["背景"]->getwidth(),
		Res::GetInstance()->img["背景"]->getheight());
	mciSendString("open ./res/game_music.mp3", 0, 0, 0); //在窗口創建出來後添加音樂
	mciSendString("play ./res/game_music.mp3 repeat", 0, 0, 0); //重復播放
}
 
Graph::~Graph()
{
	closegraph();    //對象沒瞭就關閉窗口
}
 
void Graph::DrawMap()
{
	Res::DrawIMG(0, 0, "背景");
}

time.hpp – – – 定義和實現寫一起,用hpp做結尾      用時間控制子彈的發射 – – – 定時器

#pragma once
#include "common.h"
class MyTimer 
{
public:
	static bool Timer(int duration, int id)     //時間間隔    id
	{
		static int startTime[10];               //開始時間---靜態變量自動初始化為0
		int endTime = clock();                  //結束時間
		if (endTime - startTime[id] >= duration)//結束時間-開始時間>=時間間隔
		{
			startTime[id] = endTime;            //把原來的結束時間改為新的開始時間
			return true;
		}
		return false;
	}
 
};

enemy.h        敵機

#pragma once
#include "common.h"
class Element;
class Enemy
{
public:
	Enemy();
	~Enemy();
	void DrawEnemy(int preIndex);  //畫第幾張
	void MoveEnemy(int speed);
	Element* createEnemy();        //創建敵機
	list<Element*>& GetEnemy();    //訪問敵機---需要做碰撞檢測
protected:
	list<Element*> enemyPlane;     //(存儲)多個敵機
};

enemy.cpp

#include "enemy.h"
#include "element.h"
#include "res.h"
Enemy::Enemy()
{
 
}
Enemy::~Enemy()
{
 
}
void Enemy::DrawEnemy(int preIndex)
{
    for (auto v : enemyPlane) //畫元素
    {
        if (v->GetLive())     //判斷敵機是否存在
        {
            v->DrawElement(preIndex);
        }
    }
}
void Enemy::MoveEnemy(int speed)
{
    for (auto v : enemyPlane) 
    {
        v->MoveElement(speed, Point::down);    //速度    方向
    }
}
Element* Enemy::createEnemy()  //獲取窗口寬高---隨機x,y坐標 從窗口上邊界出來 速度 血量
{
    return new Element(rand()%Res::GetInstance()->WIDTH("背景"),
        -1*Res::GetInstance()->HEIGHT("敵機"),"敵機",1, 3); 
}
list<Element*>& Enemy::GetEnemy()
{
    // TODO: 在此處插入 return 語句
    return enemyPlane;    //返回一個容器
}

采用多線程的方式播放音樂

到此這篇關於c++使用Easyx圖形庫實現飛機大戰的文章就介紹到這瞭。希望對大傢的學習有所幫助,也希望大傢多多支持WalkonNet。

推薦閱讀: