跨年烟花代码:
1.说明:
上面这张gif图就是效果图片咯 大家可以看看
完整视频我会在后续的博客中记录滴(当然也可以去我的小红书观看: 大板栗)
下面就是代码了
#include<graphics.h>
#include<time.h>
#include<stdlib.h>
#include<math.h>
#include<conio.h>
#include<windows.h>
#define PI 3.1415925
#pragma comment(lib,"winmm.lib")//就是引入静态库winmm.lib
/*#pragma 是一条编译器指令,是给告诉编译器你要链接一些东西,
然后在后面的comment里面指明是什么东西*/
#define NUM 15 //烟花弹个数,烟花
//烟花弹
struct jet
{
int x, y; //烟花弹坐标
int hx, hy; //烟花弹最高点坐标
bool shoot; //烟花弹是否处于发射状态
DWORD t1, t2, dt; //发射时间 引爆时间 间隔时间
IMAGE img[2]; //2张图片 一明一暗 01下标
byte n : 1; //C结构体 位段 //n 变量 1个位 0,1 n++ 0,1,0,1
}jet[NUM]; //烟花弹个数
//烟花
struct Fire
{
int x, y; //烟花的坐标
int r; //烟花的半径
int max_r; //烟花的最大半径
int cen_x, cen_y; //中心距左上角的距离
int width, height; //长宽
int xy[240][240]; //重要,像素,矩阵
bool draw; //画出
bool show; //显示
DWORD t1, t2, dt; //发射时间 引爆时间 间隔时间
}fire[NUM];
//初始化函数
void FireInit(int i)
{
//初始化烟花弹
jet[i].t1 = GetTickCount(); //GetTickCount()返回从操作系统启动到当前所经过的毫秒数。使用前包含windows.h。
jet[i].shoot = false; //未发射
jet[i].dt = 10; //上升时间
jet[i].n = 0; //
//初始化烟花
fire[i].show = false; //未引爆
fire[i].r = 0;
fire[i].dt = 5; //上升时间
fire[i].t1 = GetTickCount();
fire[i].max_r = rand() % 50 + 100; //100-149
fire[i].cen_x = rand() % 30 + 80; //中心距左上角的距离
fire[i].cen_y = rand() % 30 + 80; //
fire[i].width = 240; //宽
fire[i].height = 240; //长
}
//加载
void Load()
{
//加载烟花弹
IMAGE jetimg;
loadimage(&jetimg, L"./fire/shoot.jpg", 200, 50);
SetWorkingImage(&jetimg);
for (int i = 0; i < NUM; i++)
{
int n = rand() % 10; //01234
getimage(&jet[i].img[0], n * 20, 0, 20, 50);
getimage(&jet[i].img[1], (n + 5) * 20, 0, 20, 50);
}
SetWorkingImage(NULL);
//加载烟花
IMAGE fireimage, Fireimage;
loadimage(&Fireimage, L"./fire/flower.jpg", 3120, 240);
for (int i = 0; i < NUM; i++)
{
SetWorkingImage(&Fireimage);
getimage(&fireimage, i * 240, 0, 240, 240);
SetWorkingImage(&fireimage);
for (int a = 0; a < 240; a++)
{
for (int b = 0; b < 240; b++)
{
fire[i].xy[a][b] = getpixel(a, b);
}
}
}
SetWorkingImage(NULL);
}
//选择烟花弹
void ChoiceJet(DWORD& t1)
{
DWORD t2 = GetTickCount();
if (t2 - t1 > 50) //烟花弹出现的时间间隔100ms
{
//烟花弹个数
int i = rand() % 15;
//不处于发射状态
if (jet[i].shoot == false && fire[i].show == false)
{
//烟花弹
jet[i].x = rand() % 2000;
jet[i].y = rand() % 800 + 600; //450-549
jet[i].hx = jet[i].x;
jet[i].hy = rand() % 300; //0-299
jet[i].shoot = true; //发射状态
putimage(jet[i].x, jet[i].y, &jet[i].img[jet[i].n], SRCINVERT);
}
t1 = t2;
}
}
//判断发射
void Shoot()
{
for (int i = 0; i < NUM; i++)
{
jet[i].t2 = GetTickCount();
if (jet[i].t2 - jet[i].t1 >= jet[i].dt && jet[i].shoot == true)
{
putimage(jet[i].x, jet[i].y, &jet[i].img[jet[i].n], SRCINVERT);
if (jet[i].y >= jet[i].hy)
{
jet[i].n++; //闪烁
jet[i].y -= 5;
}
putimage(jet[i].x, jet[i].y, &jet[i].img[jet[i].n], SRCINVERT);
if (jet[i].y <= jet[i].hy)
{
putimage(jet[i].x, jet[i].y, &jet[i].img[jet[i].n], SRCINVERT);
jet[i].shoot = false;
//达到最大高度,接下来交给烟花
//重新发射
fire[i].x = jet[i].hx;
fire[i].y = jet[i].hy;
fire[i].show = true;
}
}
jet[i].t1 = jet[i].t2;
}
}
//显示烟花
void ShowFire(DWORD* pMem)
{
int drt[16] = { 5,5,5,5,5,10,25,25,25,25,55,55,55,55,55,65 };
for (int i = 0; i < NUM; i++)
{
fire[i].t2 = GetTickCount();
if (fire[i].t2 - fire[i].t1 >= fire[i].dt && fire[i].show == true)
{
if (fire[i].r < fire[i].max_r)
{
fire[i].r++;
fire[i].dt = drt[fire[i].r / 10];
fire[i].draw = true;
}
if (fire[i].r >= fire[i].max_r - 1)
{
fire[i].draw = false;
FireInit(i);
}
fire[i].t1 = fire[i].t2;
// 如果该号炮花可爆炸,根据当前爆炸半径画烟花,颜色值接近黑色的不输出。
if (fire[i].draw)
{
for (double a = 0; a <= 6.28; a += 0.01)
{
int x1 = (int)(fire[i].cen_x + fire[i].r * cos(a));
int y1 = (int)(fire[i].cen_y - fire[i].r * sin(a));
if (x1 > 0 && x1 < fire[i].width && y1>0 && y1 < fire[i].height)
{
int b = fire[i].xy[x1][y1] & 0xff;
int g = (fire[i].xy[x1][y1] >> 8) & 0xff;
int r = (fire[i].xy[x1][y1] >> 16);
// 烟花像素点在窗口上的坐标
int xx = (int)(fire[i].x + fire[i].r * cos(a));
int yy = (int)(fire[i].y - fire[i].r * sin(a));
//较暗的像素点不输出、防止越界
if (r > 0x20 && g > 0x20 && b > 0x20 && xx > 0 && xx < 2000 && yy >0 && yy < 1000)
{
pMem[yy * 2000 + xx] = BGR(fire[i].xy[x1][y1]);
}
fire[i].draw = false;
}
}
}
}
}
}
void welcome() //这里是文字编写界面哦 大家如果想要修改文字的话 只需要把自己的话修改上去就好啦
{
setcolor(RGB(233,235,254));
Sleep(700);
for (int i = 0; i < 50; i++)
{
int x = 600 + int(180 * sin(PI * 2 * i / 60));
int y = 250 + int(180 * cos(PI * 2 * i / 60));
cleardevice();
IMAGE image;
loadimage(&image, L"./fire/xinxin1.png", 2000, 1005);
putimage(0, 95, &image);
settextstyle(i, 0, L"楷体");
outtextxy(x +300, y+100 , L"祝屏幕前的小红薯们 ");
outtextxy(x +300, y + 200, L" 新年快乐!");
Sleep(80);
}
Sleep(130);
cleardevice();
IMAGE image;
loadimage(&image, L"./fire/xinxin1.png", 2000, 1005);
putimage(0, 95, &image);
settextstyle(25, 0, L"楷体");
outtextxy(400, 150, L"Dear 小红薯们:");
outtextxy(700, 250, L"大家晚上好呀");
outtextxy(700, 300, L"这是今年的最后一份笔记");
outtextxy(700, 350, L"自己也是第一次用视频的方式来展现的");
outtextxy(700, 400, L"然后就是希望在新的一年");
outtextxy(700, 450, L"自己能以更好的状态");
outtextxy(700, 500, L"去分享更多有用的东西给大家");
outtextxy(700, 550, L"最后祝屏幕前的小红薯们");
outtextxy(700, 600, L"所求皆如愿");
outtextxy(700, 650, L"所行化坦途");
outtextxy(700, 700, L"新年快乐呀!");
outtextxy(700, 750, L"");
outtextxy(1300, 900, L"大板栗");
outtextxy(1300, 950, L"2021年12月30日");
Sleep(4000);
cleardevice();
}
//主函数
int main()
{
//初始界面(1200,800)
initgraph(2000, 1100);
//初始化种子
srand((unsigned int)time(NULL));
//祝大家新年快乐呀!
mciSendString(L"open ./fire/dabanli.mp3 alias music", 0, 0, 0); //send(发送) string(字符串)
mciSendString(L"play music", 0, 0, 0);
//其它音乐类型 wav PlaySound()
//0,0,0 音乐播放器时:播放设备,快进设备 快退 暂停
welcome();
IMAGE image;
loadimage(&image, L"./fire/xinxin1.png", 2000, 1005);
putimage(0, 95, &image);
DWORD t1 = GetTickCount();
DWORD* pMem = GetImageBuffer();
for (int i = 0; i < NUM; i++)
{
FireInit(i);
}
Load();
BeginBatchDraw();
while (1)
{
// 随机选择像素点擦除
for (int clr = 0; clr < 200; clr++)
{
int px1 = rand() % 2000;
int py1 = rand() % 1100;
// 防止越界
if (py1 < 1099)
{
//对显存赋值擦除像素点
pMem[py1 * 1000 + px1] = pMem[py1 * 1000 + px1 + 1] = BLACK;
}
}
ChoiceJet(t1);
Shoot();
ShowFire(pMem);
FlushBatchDraw();
}
/*
putimage(0, 0, &jet[3].img[0]);
putimage(100, 0, &jet[3].img[1]);
*/
closegraph();
return 0;
}
/*
//图片
//定义一张图片
IMAGE img;
//加载一张图片
loadimage(&img, L"./fire/shoot.jpg", 200, 50);
//贴图
putimage(400, 550, &img);
//获取单个图片
IMAGE img1;
SetWorkingImage(&img); //重哪里获取:img
getimage(&img1, 180, 0, 20, 50); //获取一个
SetWorkingImage(NULL); //拿完了
putimage(0, 0, &img1); //贴图试试
*/
2.链接
这是源码和文件的提取地址 大家感兴趣的话可以自己提取哦
链接:https://pan.baidu.com/s/1Vx1wpaDqRQA9GdBfNc3OMw
提取码:b0ik
最后就是说明了 当然里面也有说明书