当前位置:网站首页>图形学-粒子系统 (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;
}
原网站

版权声明
本文为[@Moota]所创,转载请带上原文链接,感谢
https://blog.csdn.net/m0_51819222/article/details/126128632