当前位置:网站首页>图形学-粒子系统 (Particle System)
图形学-粒子系统 (Particle System)
2022-08-03 14:03:00 【@Moota】
一. 说明
使用 C++ 利用 EasyX库 实现, 在 Krissi 大佬 代码基础上,修改和添加一些功能。
主要优点:
- 能模拟一些简单的粒子行为。
主要缺点:
- 功能过于简陋。
- 公式没有参考严谨的物理与数学。
- 代码超级耦合,未能设计接口提供扩展,只能在源代码上修改。
本人代码萌新,虽然有注释,但是存在不明白或错误之处,欢迎大家一起讨论和交流。
二. 截图
三. 代码
C++ 代码有两个文件,ParticleSystem.h 和 ParticleSystem.cpp
ParticleSystem.h
/** * @name 粒子系统(Particle System) * @code Windows10,Rider 2022 | Visual Studio 2022,EasyX for VisualStudio2022 * @origin "https://codebus.cn/zhaoh/liquid-particles" * @author Moota <[email protected]> * @date 2022 - 08 - 02 */
/** * 注意事项: * @note 代码是在 Krissi 大神的基础上修改!!! * * 特效添加步骤: * @step 添加特效函数,在:FParticleSystem。 * 如: OnEffectGravity(float InDeltaTime, FParticle& InParticle) * * @step 添加控制信号,在:FParticleSystem。 * 如: bool IsEffectGravity; * * @step 添加控制按键,在:FParticleSystem::HandleInput。 * 如: case 'P': IsEffectGravity =! IsEffectGravity; * * @step 添加提示信息。在:FParticleSystem::ShowHelp。 * 如: "[P]: Effect - Gravity ON"; * * 其他 * @note 虽然不能全屏,当你拖动窗口到略微超过屏幕,系统会自动帮你上下对齐窗口,就相当于全屏啦。 * @note 为了简单起见,很多设计限于两个文件没有实现,不过还好啦。 * @note 萌新初学,欢迎一起讨论和交流!!! */
/** * 不算 Bug 的 Bug: * @bug 鼠标以按下状态超过窗口范围,再以松开状态返回窗口时,鼠标原作用会保持,比如会一直集聚粒子。 * @reason 原因是超过窗口范围的鼠标信息未被利用,再次点击就会恢复正常。 * @note 不过这样也很好看,某种意义上还很方便呃。 */
#pragma once
#include <ctime>
#include <graphics.h>
#include <vector>
/** * 窗口管理器 */
class FWindowManager
{
public:
FWindowManager(int InWidth, int InHeight)
{
// 创建绘图窗口
initgraph(InWidth, InHeight);
// 启用批绘图模式
BeginBatchDraw();
}
~FWindowManager()
{
// 关闭批绘图模式
EndBatchDraw();
// 关闭绘图窗口
closegraph();
}
inline void Flush() const
{
// 显示缓存的绘制内容
FlushBatchDraw();
}
};
/** * 粒子结构 */
struct FParticle
{
public:
COLORREF Color; // 颜色
float PosX, PosY; // 坐标
float VelX, VelY; // 速度
float Friction; // 摩擦力
float Radius; // 半径
float LifeTime; // 生命周期
public:
float NextX, NextY; // 下一次坐标
bool IsPendingKilled; // 是否进入死亡
FParticle(COLORREF InColor = RED, float InPosX = 0.0f, float InPosY = 0.0f, float InVelX = 0.0f, float InVelY = 0.0f, float InFriction = 0.96f, float InRadius = 0.4f, float InLifeTime = 0.0f)
: Color(InColor), PosX(InPosX), PosY(InPosY), VelX(InVelX), VelY(InVelY), Friction(InFriction), Radius(InRadius), LifeTime(InLifeTime)
{
NextX = 0;
NextY = 0;
IsPendingKilled = false;
}
};
/** * 二维坐标 */
struct FVector2D
{
float X, Y;
};
/** * 漩涡信息 */
struct FVortex
{
FVector2D Position;
float Radius;
float RotateSpeed;
};
/** * 粒子系统 */
class FParticleSystem
{
public:
FParticleSystem(float InWidth = 1540.0f, float InHeight = 840.0f, int InParticlesNum = 666);
/** * 运行粒子系统 */
void Play();
public:
inline void SetDeltaTime(const float InDeltaTime)
{
CustomDeltaTime = InDeltaTime;
}
inline float GetDeltaTime() const
{
return CustomDeltaTime;
}
private:
/** * 初始化 */
void Initialize();
/** * 处理输入 */
void HandleInput();
/** * 显示帮助信息 */
void ShowHelp();
/** * 绘制动画(一帧) */
void Render(const float InDeltaTime);
/** * 计算 FPS */
float CalculateFps(const float InNewTime);
/** * 计算增量时间 */
float CalculateDeltaTime(const clock_t InCurTime);
/** * 绝对延时 */
static void Delay(const clock_t Ms);
/** * 帧率限制 */
float LimitFps(const float InElapsedTime);
private:
/** * 鼠标范围特效:点击会出现小圆表示落点 */
void OnEffectMouseField(const float InDeltaTime) const;
/** * 粒子轨迹消失特效:开启将不会记录粒子移动轨迹 */
void OnEffectDisappearTrack(const float InDeltaTime) const;
/** * 粒子边界特效:碰到边界反弹 */
void OnEffectReboundWall(const float InDeltaTime, FParticle& InParticle) const;
/** * 粒子大小特效:速度越大,粒子半径越大 */
void OnEffectRadiusAboutSpeed(const float InDeltaTime, FParticle& InParticle) const;
/** * 粒子吹散特效:鼠标右击时,以落点吹散粒子 */
void OnEffectBlow(const float InDeltaTime, FParticle& InParticle) const;
/** * 粒子聚集特效:鼠标左击时,以落点聚集粒子 */
void OnEffectGather(const float InDeltaTime, FParticle& InParticle) const;
/** * 粒子搅拌特效:鼠标移动时,以落点搅动粒子 */
void OnEffectStir(const float InDeltaTime, FParticle& InParticle) const;
/** * 粒子摩擦力特效:粒子移动受摩擦力 */
void OnEffectFriction(const float InDeltaTime, FParticle& InParticle) const;
/** * 粒子漩涡特效:空间存在漩涡,作用于粒子 */
void OnEffectVortex(const float InDeltaTime, FParticle& InParticle) const;
/** * 粒子随机微动特效:不受作用的情况下,粒子会随机移动 */
void OnEffectRandomMove(const float InDeltaTime, FParticle& InParticle) const;
/** * 粒子随滚轮缩放特效:鼠标滚轮移动控制粒子显示大小 */
void OnEffectScaleWithWheel(const float InDeltaTime, FParticle& InParticle) const;
/** * 重力特效:粒子受重力影响 */
void OnEffectGravity(const float InDeltaTime, FParticle& InParticle) const;
/** * 粒子死亡特效:粒子也会消逝啊 */
void OnEffectDeath(const float InDeltaTime, FParticle& InParticle) const;
private:
const float Width; // 屏幕宽度
const float Height; // 屏幕高度
const int ParticlesNum; // 粒子数量
std::vector<FParticle> Particles; // 粒子数组
int CurMousePosX = 0, CurMousePosY = 0; // 当前鼠标坐标
int PrevMousePosX = 0, PrevMousePosY = 0; // 上次鼠标坐标
int MouseVelX = 0, MouseVelY = 0; // 鼠标速度
bool IsLeftMouseDown = false; // 是否按下鼠标左键
bool IsRightMouseDown = false; // 是否按下鼠标右键
DWORD* GBuffer = nullptr; // 显示缓冲区指针
float CustomHighFps = 90.0f; // 限制高帧率
float CustomLowFps = 10.0f; // 限制低帧率
float CustomDeltaTime = 1 / 90.0f; // 增量时间
float GraphicsFps = 0.0f; // 预计帧率
float GraphicsTime = 0.0f; // 每帧时间
FWindowManager WindowManager; // 窗口管理器
private:
float RandomDis; // 进入随机距离
float BlowDis; // 进入扩散距离
float GatherDis; // 进入聚集距离
float StirDis; // 进入搅拌距离
private:
bool IsShowHelp = true; // 是否显示帮助
bool IsContinueGraphics = true; // 是否继续系统
bool IsContinueSimulate = true; // 是否继续模拟
mutable float ScaleSize = 1.0f; // 滚轮缩放大小
private:
// 特效控制,顾名思义
bool IsEffectMouseField = false;
bool IsEffectReboundWall = true;
bool IsEffectRadiusAboutSpeed = true;
bool IsEffectBlow = true;
bool IsEffectGather = true;
bool IsEffectStir = true;
bool IsEffectFriction = true;
bool IsEffectVortex = true;
bool IsEffectRandomMove = true;
bool IsEffectScaleWithWheel = true;
bool IsEffectDisappearTrack = true;
bool IsEffectGravity = false;
bool IsEffectDeath = false;
};
ParticleSystem.cpp
#include "ParticleSystem.h"
#include <ctime>
#include <iostream>
#include <string>
FParticleSystem::FParticleSystem(float InWidth, float InHeight, int InParticlesNum)
: Width(InWidth), Height(InHeight), ParticlesNum(InParticlesNum), WindowManager(static_cast<int>(Width), static_cast<int>(Height))
{
// 初始化鼠标变量
CurMousePosX = PrevMousePosX = static_cast<int>(InWidth) / 2;
CurMousePosY = PrevMousePosY = static_cast<int>(InWidth) / 2;
RandomDis = InWidth;
BlowDis = InWidth * 0.5f;
GatherDis = InWidth * 0.86f;
StirDis = InWidth * 0.125f;
CustomDeltaTime = 1 / CustomHighFps;
}
void FParticleSystem::Play()
{
Initialize();
static clock_t OldTime;
static clock_t NewTime;
while (IsContinueGraphics)
{
OldTime = clock();
// 计算增量时间
CalculateDeltaTime(clock());
// 处理输入消息
HandleInput();
// 绘制所需图形
Render(GetDeltaTime());
// 显示缓存绘制
WindowManager.Flush();
NewTime = clock();
// 限制帧数并计算
CalculateFps(LimitFps(static_cast<float>(NewTime - OldTime)));
}
}
void FParticleSystem::Initialize()
{
// 设置随机种子
srand(static_cast<unsigned>(time(nullptr) + 5201314));
// 初始化粒子数组
Particles.reserve(ParticlesNum);
for (int i = 0; i < ParticlesNum; ++i)
{
Particles.emplace_back(RGB(rand() % 256, rand() % 256, rand() % 256), Width * 0.5f, Height * 0.5f, cos(static_cast<float>(i)) * (rand() % 34), sin(static_cast<float>(i)) * (rand() % 34),
0.96f, 0.4f, 2.0f + rand() % 100);
}
// 获取显示缓冲区指针
GBuffer = GetImageBuffer(nullptr);
}
void FParticleSystem::HandleInput()
{
ExMessage Input{
};
while (peekmessage(&Input, EM_MOUSE | EM_KEY))
{
switch (Input.message)
{
case WM_MOUSEMOVE:
CurMousePosX = Input.x;
CurMousePosY = Input.y;
break;
case WM_MOUSEWHEEL:
ScaleSize = min(max(ScaleSize + Input.wheel / 500.0f, 0.5f), 5.0f);
break;
case WM_LBUTTONDOWN:
IsLeftMouseDown = true;
break;
case WM_LBUTTONUP:
IsLeftMouseDown = false;
break;
case WM_RBUTTONDOWN:
IsRightMouseDown = true;
break;
case WM_RBUTTONUP:
IsRightMouseDown = false;
break;
case WM_KEYDOWN:
{
switch (Input.vkcode)
{
case VK_ESCAPE:
IsContinueGraphics = false;
break;
case VK_SPACE:
IsContinueSimulate = !IsContinueSimulate;
break;
case VK_TAB:
IsShowHelp = !IsShowHelp;
break;
case 'A':
case 'a':
IsEffectMouseField = !IsEffectMouseField;
break;
case 'B':
case 'b':
IsEffectReboundWall = !IsEffectReboundWall;
break;
case 'C':
case 'c':
IsEffectRadiusAboutSpeed = !IsEffectRadiusAboutSpeed;
break;
case 'D':
case 'd':
IsEffectBlow = !IsEffectBlow;
break;
case 'E':
case 'e':
IsEffectGather = !IsEffectGather;
break;
case 'F':
case 'f':
IsEffectStir = !IsEffectStir;
break;
case 'G':
case 'g':
IsEffectFriction = !IsEffectFriction;
break;
case 'H':
case 'h':
IsEffectVortex = !IsEffectVortex;
break;
case 'I':
case 'i':
IsEffectRandomMove = !IsEffectRandomMove;
break;
case 'J':
case 'j':
IsEffectScaleWithWheel = !IsEffectScaleWithWheel;
break;
case 'K':
case 'k':
IsEffectDisappearTrack = !IsEffectDisappearTrack;
break;
case 'L':
case 'l':
IsEffectGravity = !IsEffectGravity;
break;
case 'M':
case 'm':
IsEffectDeath = !IsEffectDeath;
break;
default:
break;
}
break;
}
default:
break;
}
}
}
void FParticleSystem::ShowHelp()
{
if (IsShowHelp)
{
static wchar_t Message[255];
const int TextHeight = 16;
int Lines = -1;
swprintf_s(Message, _T("\t\tfps: %.1f"), GraphicsFps);
outtextxy(0, TextHeight * (++Lines), Message);
swprintf_s(Message, _T("\t\ttime: %.1fms"), GraphicsTime);
outtextxy(0, TextHeight * (++Lines), Message);
swprintf_s(Message, _T("\t\tparticles: %d / %d"), Particles.size(), ParticlesNum);
outtextxy(0, TextHeight * (++Lines), Message);
(++Lines);
outtextxy(0, TextHeight * (++Lines), _T("\t\tMouse"));
outtextxy(0, TextHeight * (++Lines), _T("\t\tMove: Stir"));
outtextxy(0, TextHeight * (++Lines), _T("\t\tLeft: Gather"));
outtextxy(0, TextHeight * (++Lines), _T("\t\tRight: Blow"));
outtextxy(0, TextHeight * (++Lines), _T("\t\tWheel: Scale"));
(++Lines);
outtextxy(0, TextHeight * (++Lines), _T("\t\tKeyword"));
outtextxy(0, TextHeight * (++Lines), _T("\t\t[Esc]: Exit System"));
outtextxy(0, TextHeight * (++Lines), _T("\t\t[Tab]: Hide Status"));
outtextxy(0, TextHeight * (++Lines), _T("\t\t[Space]: Pause System"));
swprintf_s(Message, _T("\t\t[A]: Effect - MouseField %s"), IsEffectMouseField == true ? _T("ON") : _T("OFF"));
outtextxy(0, TextHeight * (++Lines), Message);
swprintf_s(Message, _T("\t\t[B]: Effect - ReboundWall %s"), IsEffectReboundWall == true ? _T("ON") : _T("OFF"));
outtextxy(0, TextHeight * (++Lines), Message);
swprintf_s(Message, _T("\t\t[C]: Effect - RadiusAboutSpeed %s"), IsEffectRadiusAboutSpeed == true ? _T("ON") : _T("OFF"));
outtextxy(0, TextHeight * (++Lines), Message);
swprintf_s(Message, _T("\t\t[D]: Effect - Blow %s"), IsEffectBlow == true ? _T("ON") : _T("OFF"));
outtextxy(0, TextHeight * (++Lines), Message);
swprintf_s(Message, _T("\t\t[E]: Effect - Gather %s"), IsEffectGather == true ? _T("ON") : _T("OFF"));
outtextxy(0, TextHeight * (++Lines), Message);
swprintf_s(Message, _T("\t\t[F]: Effect - Stir %s"), IsEffectStir == true ? _T("ON") : _T("OFF"));
outtextxy(0, TextHeight * (++Lines), Message);
swprintf_s(Message, _T("\t\t[G]: Effect - Friction %s"), IsEffectFriction == true ? _T("ON") : _T("OFF"));
outtextxy(0, TextHeight * (++Lines), Message);
swprintf_s(Message, _T("\t\t[H]: Effect - Vortex %s"), IsEffectVortex == true ? _T("ON") : _T("OFF"));
outtextxy(0, TextHeight * (++Lines), Message);
swprintf_s(Message, _T("\t\t[I]: Effect - RandomMove %s"), IsEffectRandomMove == true ? _T("ON") : _T("OFF"));
outtextxy(0, TextHeight * (++Lines), Message);
swprintf_s(Message, _T("\t\t[J]: Effect - Scale %s"), IsEffectScaleWithWheel == true ? _T("ON") : _T("OFF"));
outtextxy(0, TextHeight * (++Lines), Message);
swprintf_s(Message, _T("\t\t[K]: Effect - DisappearTrack %s"), IsEffectDisappearTrack == true ? _T("ON") : _T("OFF"));
outtextxy(0, TextHeight * (++Lines), Message);
swprintf_s(Message, _T("\t\t[L]: Effect - Gravity %s"), IsEffectGravity == true ? _T("ON") : _T("OFF"));
outtextxy(0, TextHeight * (++Lines), Message);
swprintf_s(Message, _T("\t\t[M]: Effect - Death %s"), IsEffectDeath == true ? _T("ON") : _T("OFF"));
outtextxy(0, TextHeight * (++Lines), Message);
}
}
void FParticleSystem::Render(const float InDeltaTime)
{
if (IsContinueSimulate)
{
ShowHelp();
OnEffectMouseField(InDeltaTime);
OnEffectDisappearTrack(InDeltaTime);
MouseVelX = CurMousePosX - PrevMousePosX;
MouseVelY = CurMousePosY - PrevMousePosY;
PrevMousePosX = CurMousePosX;
PrevMousePosY = CurMousePosY;
for (int i = 0; i < Particles.size(); i++)
{
FParticle Temp = Particles[i];
OnEffectDeath(InDeltaTime, Temp);
OnEffectRandomMove(InDeltaTime, Temp);
OnEffectBlow(InDeltaTime, Temp);
OnEffectGather(InDeltaTime, Temp);
OnEffectStir(InDeltaTime, Temp);
OnEffectFriction(InDeltaTime, Temp);
OnEffectRadiusAboutSpeed(InDeltaTime, Temp);
OnEffectVortex(InDeltaTime, Temp);
Temp.NextX = Temp.PosX + Temp.VelX;
Temp.NextY = Temp.PosY + Temp.VelY;
OnEffectReboundWall(InDeltaTime, Temp);
OnEffectGravity(InDeltaTime, Temp);
Temp.PosX = Temp.NextX;
Temp.PosY = Temp.NextY;
OnEffectScaleWithWheel(InDeltaTime, Temp);
OnEffectDeath(InDeltaTime, Temp);
Particles[i] = Temp;
// 画小球
if (Particles[i].IsPendingKilled == true)
{
std::swap(Particles[i], Particles[Particles.size() - 1]);
Particles.pop_back();
}
else
{
setfillcolor(Particles[i].Color);
solidcircle(static_cast<int>(Particles[i].NextX + 0.5f), static_cast<int>(Particles[i].NextY + 0.5f), static_cast<int>(ScaleSize * Particles[i].Radius + 0.5f));
}
}
}
}
float FParticleSystem::CalculateFps(const float InElapsedTime)
{
static const int FpsCount = 8; // 每 8 次计算一次帧数
static int Count = 0;
static float TotalElapsedTime = 0;
++Count;
GraphicsTime = InElapsedTime;
TotalElapsedTime += GraphicsTime;
if (Count >= FpsCount)
{
GraphicsFps = static_cast<float>(Count) / (TotalElapsedTime / 1000.0f);
Count = 0;
TotalElapsedTime = 0;
}
return GraphicsFps;
}
float FParticleSystem::CalculateDeltaTime(const clock_t InCurTime)
{
static clock_t LastTime = clock();
static const float LowLimit = 1.0f / CustomHighFps;
static const float HighLimit = 1.0f / CustomLowFps;
float CurDeltaTime = (InCurTime - LastTime) / 1000.0f;
if (CurDeltaTime < LowLimit)
{
CurDeltaTime = LowLimit;
}
if (CurDeltaTime > HighLimit)
{
CurDeltaTime = HighLimit;
}
SetDeltaTime(CurDeltaTime);
LastTime = InCurTime;
return CurDeltaTime;
}
void FParticleSystem::Delay(const clock_t Ms)
{
static clock_t OldTime = clock();
OldTime = clock();
// 不清楚是否有效
while ((clock() - OldTime) < Ms)
{
Sleep(1);
}
}
float FParticleSystem::LimitFps(const float InElapsedTime)
{
float OffsetTime = (1000.0f / CustomHighFps) - InElapsedTime;
if (OffsetTime >= 0.01f)
{
Delay(OffsetTime + 1);
}
else
{
OffsetTime = 0.0f;
}
return InElapsedTime + OffsetTime;
}
void FParticleSystem::OnEffectMouseField(const float InDeltaTime) const
{
if (IsEffectMouseField)
{
static float Radius = 2.0f;
static float Times = 1.5f;
static float MouseFieldSpeed = 1.0f;
static COLORREF Color = RGB(187, 34, 128);
if (IsLeftMouseDown)
{
Times = 1.5f;
if (Times >= 0.0f)
{
Times -= MouseFieldSpeed * InDeltaTime;
Radius += MouseFieldSpeed * InDeltaTime;
setfillcolor(Color);
setlinecolor(Color);
fillcircle(CurMousePosX + MouseVelX * rand() % 2 * 0.01f, CurMousePosY + MouseVelY * rand() % 2 * 0.01f, max(Radius, 5.0f));
}
}
else
{
if (Times > 0.0f)
{
Times -= MouseFieldSpeed * InDeltaTime;
Radius -= MouseFieldSpeed * InDeltaTime;
setfillcolor(Color);
setlinecolor(Color);
fillcircle(CurMousePosX + rand() % 2, CurMousePosY + rand() % 2, max(Radius, 2.0f));
}
else
{
Color = RGB(rand()%256, rand()%256, rand()%256);
}
}
}
}
void FParticleSystem::OnEffectDisappearTrack(const float InDeltaTime) const
{
if (IsEffectDisappearTrack)
{
for (int i = static_cast<int>(Width * Height) - 1; i >= 0; --i)
{
if (GBuffer[i] != 0)
{
GBuffer[i] = RGB(GetRValue(GBuffer[i]) >> 1, GetGValue(GBuffer[i]) >> 1, GetBValue(GBuffer[i]) >> 1);
}
}
}
}
void FParticleSystem::OnEffectReboundWall(const float InDeltaTime, FParticle& InParticle) const
{
if (IsEffectReboundWall)
{
static float ReboundLoss = 0.85f;
if (InParticle.NextX > Width)
{
InParticle.NextX = Width;
InParticle.VelX *= -1 * ReboundLoss;
}
else if (InParticle.NextX < 0)
{
InParticle.NextX = 0;
InParticle.VelX *= -1 * ReboundLoss;
}
if (InParticle.NextY > Height)
{
InParticle.NextY = Height;
InParticle.VelY *= -1 * ReboundLoss;
}
else if (InParticle.NextY < 0)
{
InParticle.NextY = 0;
InParticle.VelY *= -1 * ReboundLoss;
}
}
}
void FParticleSystem::OnEffectRadiusAboutSpeed(const float InDeltaTime, FParticle& InParticle) const
{
if (IsEffectRadiusAboutSpeed)
{
const float AvgVx = fabs(InParticle.VelX);
const float AvgVy = fabs(InParticle.VelY);
const float AvgV = (AvgVx + AvgVy) * 0.5f;
InParticle.Radius = AvgV * 0.45f;
InParticle.Radius = max(min(InParticle.Radius, 1.66f), 0.4f);
}
}
void FParticleSystem::OnEffectBlow(const float InDeltaTime, FParticle& InParticle) const
{
if (IsEffectBlow)
{
float DisX = InParticle.PosX - CurMousePosX;
float DisY = InParticle.PosY - CurMousePosY;
const float Radius = sqrt(DisX * DisX + DisY * DisY);
DisX = Radius > 0.01f ? DisX / Radius : 0;
DisY = Radius > 0.01f ? DisY / Radius : 0;
if (IsRightMouseDown && Radius < BlowDis)
{
static float BlowSpeed = 10.0f;
const float BlowAcc = (1 - (Radius / BlowDis)) * 14;
InParticle.VelX += DisX * BlowAcc * BlowSpeed * InDeltaTime + 0.5f - static_cast<float>(rand()) / RAND_MAX;
InParticle.VelY += DisY * BlowAcc * BlowSpeed * InDeltaTime + 0.5f - static_cast<float>(rand()) / RAND_MAX;
}
}
}
void FParticleSystem::OnEffectGather(const float InDeltaTime, FParticle& InParticle) const
{
if (IsEffectGather)
{
float DisX = InParticle.PosX - CurMousePosX;
float DisY = InParticle.PosY - CurMousePosY;
const float Radius = sqrt(DisX * DisX + DisY * DisY);
DisX = Radius > 0.01f ? DisX / Radius : 0;
DisY = Radius > 0.01f ? DisY / Radius : 0;
if (IsLeftMouseDown && Radius < GatherDis)
{
static float GatherSpeed = 36.0f;
const float GatherAcc = (1 - (Radius / GatherDis)) * Width * 0.0014f;
InParticle.VelX -= DisX * GatherAcc * GatherSpeed * InDeltaTime;
InParticle.VelY -= DisY * GatherAcc * GatherSpeed * InDeltaTime;
}
}
}
void FParticleSystem::OnEffectStir(const float InDeltaTime, FParticle& InParticle) const
{
if (IsEffectStir)
{
const float DisX = InParticle.PosX - CurMousePosX;
const float DisY = InParticle.PosY - CurMousePosY;
const float Radius = sqrt(DisX * DisX + DisY * DisY);
static float StirSpeed = 50.0f;
if (!IsLeftMouseDown && !IsRightMouseDown && Radius < StirDis)
{
const float StirAcc = (1 - (Radius / StirDis)) * Width * 0.00026f;
InParticle.VelX += MouseVelX * StirAcc * StirSpeed * InDeltaTime;
InParticle.VelY += MouseVelY * StirAcc * StirSpeed * InDeltaTime;
}
}
}
void FParticleSystem::OnEffectFriction(const float InDeltaTime, FParticle& InParticle) const
{
if (IsEffectFriction)
{
InParticle.VelX *= InParticle.Friction;
InParticle.VelY *= InParticle.Friction;
}
}
void FParticleSystem::OnEffectVortex(const float InDeltaTime, FParticle& InParticle) const
{
if (IsEffectVortex)
{
static std::vector<FVortex> Vortexes;
const int VortexesNum = 3;
if (Vortexes.empty())
{
for (int i = 0; i < VortexesNum; ++i)
{
float PosX = 10 + rand() % static_cast<int>(Width);
float PosY = 10 + rand() % static_cast<int>(Height);
float Radius = max(30, 30 + rand()%30);
float Speed = 20.0f + rand() % 50;
if ((Width - PosX) <= 10)
{
PosX = Width - 20;
Radius = 15;
}
if ((Height - PosY) <= 10)
{
PosY = Height - 20;
Radius = 15;
}
Vortexes.push_back({
{
PosX, PosY}, Radius, Speed});
}
}
else
{
for (int i = 0; i < VortexesNum; ++i)
{
const float Vx = InParticle.PosX - Vortexes[i].Position.X - 1;
const float Vy = InParticle.PosY - Vortexes[i].Position.Y - 1;
const float Distance = sqrt(Vx * Vx + Vy * Vy);
static float RotationSpeed = 35.0f + rand() % 10;
if (Distance <= Vortexes[i].Radius)
{
// 不学习数学是这样的
// InParticle.PosX = Vortexes[i].Position.X + Distance * cos(acos(Vx / Distance)*114.6 + Rotate);
// InParticle.PosY = Vortexes[i].Position.Y + Distance * sin(asin(Vy / Distance)*114.6 + Rotate);
InParticle.PosX = Vortexes[i].Position.X + Vx * cos(RotationSpeed * InDeltaTime) + Vy * sin(Vortexes[i].RotateSpeed * InDeltaTime);
InParticle.PosY = Vortexes[i].Position.Y + Vy * cos(RotationSpeed * InDeltaTime) - Vx * sin(Vortexes[i].RotateSpeed * InDeltaTime);
if (rand() % 10 > 7)
{
InParticle.PosX += pow(-1, rand() % 2) * (rand() % 5);
InParticle.PosY += pow(-1, rand() % 2) * (rand() % 5);
}
}
}
}
}
}
void FParticleSystem::OnEffectRandomMove(const float InDeltaTime, FParticle& InParticle) const
{
if (IsEffectRandomMove)
{
float DisX = InParticle.PosX - CurMousePosX;
float DisY = InParticle.PosY - CurMousePosY;
const float Radius = sqrt(DisX * DisX + DisY * DisY);
static float RandSpeed = 100.0f;
DisX = Radius > 0.01f ? DisX / Radius : 0;
DisY = Radius > 0.01f ? DisY / Radius : 0;
const float RandomAcc = pow(-1, rand() % 2) * Radius / RandomDis * 0.5f;
InParticle.VelX += DisX * RandomAcc * RandSpeed * InDeltaTime + 0.5f - static_cast<float>(rand()) / RAND_MAX;
InParticle.VelY += DisY * RandomAcc * RandSpeed * InDeltaTime + 0.5f - static_cast<float>(rand()) / RAND_MAX;
const float AvgVx = fabs(InParticle.VelX);
const float AvgVy = fabs(InParticle.VelY);
if (AvgVx < 0.1f)
InParticle.VelX *= static_cast<float>(rand()) / RAND_MAX * 2;
if (AvgVy < 0.1f)
InParticle.VelY *= static_cast<float>(rand()) / RAND_MAX * 2;
}
}
void FParticleSystem::OnEffectScaleWithWheel(const float InDeltaTime, FParticle& InParticle) const
{
if (IsEffectScaleWithWheel)
{
// nothing to do
}
else
{
ScaleSize = 1.0f;
}
}
void FParticleSystem::OnEffectGravity(const float InDeltaTime, FParticle& InParticle) const
{
if (IsEffectGravity)
{
if ((Height - InParticle.PosY) >= 5)
{
static float GravitySpeed = 5.0f;
static float Gravity = 10.0f;
InParticle.VelY += Gravity * GravitySpeed * InDeltaTime;
}
}
}
void FParticleSystem::OnEffectDeath(const float InDeltaTime, FParticle& InParticle) const
{
if (IsEffectDeath)
{
static float GoDieSpeed = 1.0f;
InParticle.LifeTime -= GoDieSpeed * InDeltaTime;
if (InParticle.LifeTime <= 0)
{
InParticle.IsPendingKilled = true;
}
}
}
Main.cpp
#include "ParticleSystem.h"
int main()
{
FParticleSystem ParticleSystem(1540, 840, 666);
ParticleSystem.Play();
return 0;
}
边栏推荐
猜你喜欢
随机推荐
爬虫——代理搭建、爬取视频网站、爬取新闻、BeautifulSoup4介绍、bs4 遍历文档树、bs4搜索文档树、bs4使用选择器
大势,又一关乎中美竞争的关键行业,走到关键时刻了
Top 10 free proxy IP software_Domestic static IP proxy software
GMapping原理分析[通俗易懂]
如何合理安排一天,做到高效备考?
十大免费代理ip软件_国内静态ip代理软件
使用alarm函数实现sleep,使用alarm函数实现对阻塞操作设置超时
Insert or Merge
中国菜刀原理与实践
Nanoprobes Ni-NTA-Nanogold——用于 His 标签标记和检测
使用百度EasyDL实现施工人员安全装备检测
鸿湖万联扬帆富设备开发板正式合入OpenHarmony主干
参数量仅0.5B,谷歌代码补全新方法将内部生产效率提升6%
硬件业务收入下滑,为了赚钱,苹果暧昧对待流氓软件和增加广告了
线程的状态
数据分析(二)——numpy
中国手机品牌争论谁是国内第一,而它已成为中国手机在海外的代表
如何使用matlab实现分段函数「建议收藏」
STL——vector
HCIP Fifteenth Day Notes (Three-layer Architecture of Enterprise Network, VLAN and VLAN Configuration)