unity3D烟花制作-来放烟花吧!_unity particle pack-程序员宅基地

3D游戏第八次作业-烟花粒子效果实现

结果展示

先来个结果镇楼

下面是一个动画演示:

最终效果演示

本次项目使用的资源:

Unity Particle Pack:使用的Flame 01以及Twinkle等material均来自这个包,且该包有较多的示例。

粒子效果制作

我们的烟花分为以下5个部分组成:

其中,Firework主体也是一个粒子系统,它拥有下辖的5个粒子系统作为子部件。我们利用粒子系统面板中的 Sub Emitters 使得Firework所发射的每个粒子都拥有子部件的粒子效果。实际上,Firework主体只是提供一个粒子的移动,具体的粒子渲染效果全部由Firework的五个子部件来完成;

我们将从各个子部件的实现开始,逐步讲解,最后在Firework中完成各个粒子效果的合并;

trail子部件

即我们的烟花在爆炸之前飞行过程中产生的尾迹:

我们可以看到,我们的尾迹中的粒子在运动过程中发生了一些变化,总结如下:

  • size:粒子的大小先变大再变小;
  • color:粒子的颜色随着运动时间的变化而变化。
  • velocity:粒子在产生后向四周扩散,使尾迹变大,如同烟雾;

那么我们如何创建这样一个trail粒子效果呢?

  • 创建tail: 右键EffectsParticle System 创建一个新的粒子系统,然后我们给它取名为 trail

  • 设置粒子大小动态变化: 打开trail的粒子系统面板,勾选 size over lifetime 选项。并在面板下方设置粒子在运动中的size变化的曲线如下。

    通过我们的size变化的曲线,我们可以知道我们的trail所发射的粒子的大小变化为:
    缓慢变大保持一段时间大小缓慢变小瞬间变大瞬间变小
    最后一段的瞬间变大再瞬间变小是为了模拟粒子灰烬的闪动效果。

  • 设置粒子颜色动态变化: 打开trail的粒子系统面板,勾选 color over lifetime 选项。并设置颜色如下:

    最后一段颜色接近于橙红色,和粒子动态变化最后一部分的瞬间变化相结合,可以模拟出粒子灰烬的闪动效果。

  • 设置粒子运动: 打开trail的粒子系统面板,勾选 Velocity over lifetime 选项。选择粒子运动的速度变化方式为 Random between two curves

    然后令x轴和z轴方向上的速度变化曲线如下:

    而y轴方向上的速度变化不要改变,一直保持为0即可。

    这样,尾迹粒子在被产生之后就会向四周随机扩散,使得尾迹整体变大,模拟烟雾的扩散效果;

  • 设置粒子的发射方向: 打开trail的粒子系统面板,勾选 Shape 选项。并选择shape为 sphere 。这样粒子在产生的时候就会在烟花的周围随机产生,而不是直接从烟花尾部直直地产生,更加贴合现实。

  • 设置Emission和粒子系统主模板

twinkle子部件

twinkle子部件用于在烟花爆炸的时候产生一个闪光,模拟烟花爆炸时的闪光。

我们从这段动画中看到,粒子从小变大,然后瞬间变小,并在这个过程中微微旋转。因此,我们根据这个效果来设计我们的粒子系统;

步骤:

  • 创建twinkle: 右键EffectsParticle System 创建一个新的粒子系统,然后我们给它取名为 twinkle

  • 设置粒子的渲染图片为twinkle: 打开twinkle的粒子系统面板,打开 Render 选项,并改变material为Twinkle.

  • 设置粒子大小动态变化: 打开twinkle的粒子系统面板,勾选 size over lifetime 选项。并在面板下方设置粒子在运动中的size变化的曲线如下:

  • 设置粒子动态旋转: 打开twinkle的粒子系统面板,勾选 Rotation over lifetime 选项即可,不需要特意改动什么值;

  • 设置Emission和粒子系统主模板

    在这里,我们将Emission的Rate Over time,即每秒发射多少个粒子置为0,因为我们的闪光只在烟花爆炸的时候才出现,在烟花飞行的过程中不应该发射出闪光。我们转而设置Emission的Bursts,设置Bursts中的Count为1,使得Twinkle在发射粒子的时候只发射一个;Time表示发生爆炸的时候Twinkle发生一次Bursts的延迟时间。

Explore子部件

即烟花爆炸的时候产生的球型光雨:

在图中,我们可以看到球型光雨的运动特性:

  • size:类似trail的粒子,缓慢变大,再缓慢变小,之后迅速变大再迅速变小来模拟灰烬的闪烁效果;
  • color:同样类似trail的粒子,在运动过程中不断变化;
  • 发射方向:所有的粒子呈球型向四周扩散;

根据以上效果,我们就可以开始我们的设计;

步骤:

  • 创建explore: 右键EffectsParticle System 创建一个新的粒子系统,然后我们给它取名为 Explore

  • 设置粒子大小动态变化: 打开explore的粒子系统面板,勾选 size over lifetime 选项。并在面板下方设置粒子在运动中的size变化的曲线如下:

    基本上和trail的粒子的size变化一样;

  • 设置粒子颜色动态变化: 打开explore的粒子系统面板,勾选 color over lifetime 选项。并设置颜色如下:

    和trail的步骤其实一模一样,只是颜色有所不同,你可以根据自己的喜好来创建喜欢的运动渐变颜色。

  • 设置粒子的发射方向: 打开explore的粒子系统面板,勾选 Shape 选项。并选择shape为 sphere 。这样粒子就会呈球型产生,并向上下左右扩散。至于trail中为什么粒子没有呈球型扩散呢,是因为我们额外为trail中的粒子勾选了 Velocity over lifetime ,并使粒子在原来的运动的基础上再向四周扩散,从而没有产生球型。

  • 设置Emission和粒子系统主模板

    同样,我们对于球型光雨的要求也是在爆炸的时候才产生,因此Emission的Rate Over time同样置为0,而使用Bursts。和twinkle不同,光雨一次要产生很多,一次,count的数量是1000,即每个爆炸都发射1000个粒子。

Fire samll子部件

即我们飞行中的烟花:

从烟花的飞行过程中我们可以看出:

  • 烟花的尾部出现抖动;
  • 烟花本身是一个动画,即火焰存在出现到消失的过程;

根据这些,我们开始我们的制作:

  • 创建small fire: 右键EffectsParticle System 创建一个新的粒子系统,然后我们给它取名为 Fire small

  • 设置粒子的渲染图片为Flame 01: 打开Fire small的粒子系统面板,打开 Render 选项,并改变material为Flame 01.

    • Flame 01为一组火焰的图片;
  • 设置动画: 打开Fire small的粒子系统面板,打开 Texture sheet animation 选项,设置如下:

    其中Mode默认为Grid模式,不必改动。在Mode下的Tiles中的x=7, y=7表示我们将Flame 01这张图像切分为7*7的网格,网格中的每一格代表动画的一帧。我们的动画将从第一帧开始播放到最后一帧,即第49帧。其他的属性保持默认即可;这样,我们产生的粒子就会是一个自动变化的粒子。

  • 设置粒子的发射方向: 打开Fire small的粒子系统面板,勾选 Shape 选项。并选择shape为 cone ,即从锥体的底部或顶部发射粒子。设置具体如下:

    其中:

    • Angle:锥体的上底面,即发射粒子的面,相对于下底面的张开幅度,0表示和下底面的张开幅度相同。因此,在Fire small里的锥体呈圆柱形;
    • Radius:锥体的下底面的半径;

    其他均为默认值即可;但是由于锥体的上底面原本的方向不是直接指向下方的,所以我们要对粒子系统进行旋转,Fire small的transform如下:

  • 设置粒子的抖动: 可以看到我们的烟火在运动过程中是不断抖动的。这是为了模拟烟花在飞翔过程中受到气流影响造成的火焰的抖动效果,我们直接勾选 Noise 选项即可。

  • 设置粒子颜色动态变化: 打开Fire small的粒子系统面板,勾选 color over lifetime 选项。并设置颜色如下:

    因为我们原来的图片在播放动画的时候尾焰比较大,我们让动画的透明度在后面均为1,使得尾焰变小。

  • 设置Emission和粒子系统主模板

    在Fire small中的Emission,我们没有使用Rate over time和Bursts,而是使用了Rate over Distance。Rate over Distance表示每个移动距离单位发射的粒子数。也就是说,我们的Fire small在运动中才会产生粒子。

Explore2子部件

即四散的光雨效果:

可以看到,我们的Explore2实际上只是一些粒子在呈球型发散,然后每个粒子都带有我们的制作过的trail尾迹。它的实现也比较简单。trail部分和我们前面所做的trail基本相同,只是在如粒子大小以及发射粒子的数目等参数方面存在差异,之后会指出。主要的部分还是Explore2这个部件的制作;

同样的开始

  • 创建Explore2: 右键EffectsParticle System 创建一个新的粒子系统,然后我们给它取名为 Explore2

  • 设置粒子的渲染图片为Twinkle: 打开Explore2的粒子系统面板,打开 Render 选项,并改变material为Twinkle。这里复用一下Twinkle材料;

  • 设置粒子的发射方向: 打开Explore2的粒子系统面板,勾选 Shape 选项。并选择shape为 sphere ,即以球型发射粒子。其他不用改动;

  • 设置粒子动态旋转: 打开Explore2的粒子系统面板,勾选 Rotation over lifetime 选项即可,不需要特意改动什么值;

  • 设置粒子运动: 打开Explore2的粒子系统面板,勾选 Velocity over lifetime 选项。设置如下:

    我们基本不改动选项中的内容,只是改动speed Modifier一项,这一项决定了粒子的运动速度。由于粒子在爆炸的时候是以很快的速度爆发,之后快速降速,再缓慢降速,因此,我们需要在speed Modifier中体现出这种速度变化;

  • 设置粒子在运动中受重力: 打开Explore2的粒子系统面板,勾选 Force over lifetime 选项。并将z轴的力设为-20,使得粒子受到重力,在爆发后向地面坠落,模拟自然的重力;

  • 设置trail: 为了让每个粒子都拥有自己的尾迹,我们打开Explore2的粒子系统面板,勾选 Sub Emitters 选项。将一个制作完成的trail子部件加入到Explore2中,成为Explore2的子部件。然后将trail设置为Explore2子粒子系统,设置如下:

    设置trail在Explore2 Birth即开始运行的时候同时运行,这样,trail的粒子效果就能加入到每个Explore2的粒子中,产生尾迹;

  • 设置Emission和粒子系统主模板

  • trail的改动:

    • 主界面中的start size从1变为0.5;
    • 主界面中的max Particles变为20000;
    • Emission中的Rate over time变为100;

Firework主体

Firework主体的功能只是为烟花提供运动,因此不会产生任何粒子,各种粒子效果将由上面制作的各个部件来完成;

步骤:

  • 创建Firework: 右键EffectsParticle System 创建一个新的粒子系统,然后我们给它取名为 Firework

  • 将以上创建成功的组件按照开篇的组成加入到Firework中。

  • 设置粒子的发射方向: 打开Firework的粒子系统面板,勾选 Shape 选项。并选择shape为 box ,即以正方形发射粒子。由于box的默认发射方向在z轴正方向,因此,我们将Firework进行旋 转;

  • 设置粒子不渲染: 由于Firework的粒子并不需要渲染,所以,我们在 Render 选项中设置渲染模式为 None ;

  • 设置子粒子系统: 和Explore2类似,我们要勾选 Sub Emitters 选项,然后将Firework的所有子粒子系统加入到里面,设置如下:

    其中和已经介绍的Birth不同,Death是在粒子结束的时候才会触发的事件,即当粒子结束的时候,我们的Twinkle,Explore和Explore2的粒子效果会同时发生,表现为烟花爆炸;

  • 设置Emission和粒子系统主模板

  • 最终的效果:

制作放烟花场景:

代码:

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public class FireFirework : MonoBehaviour
{
    // Start is called before the first frame update
    public ParticleSystem[] firework;
    public ParticleSystem.Particle[] m_Particles;
    void Start()
    {
        GameObject[] temp;
        // 获取场景中所有的Firework,即烟花发射器;
        temp = GameObject.FindGameObjectsWithTag("firework");
        firework = new ParticleSystem[temp.Length];
        Debug.Log(temp.Length);
        // 将烟花发射器赋予firework数组;
        for(int i = 0; i < temp.Length; ++i)
        {
            firework[i] = temp[i].GetComponent<ParticleSystem>();
            firework[i].Stop();
        }
        if (m_Particles == null)
        {
            m_Particles = new ParticleSystem.Particle[firework[0].main.maxParticles];
        }
    }

    // Update is called once per frame
    [System.Obsolete]
    void Update()
    {       
        if (Input.GetButtonDown("Fire1"))
        {
            Ray ray = Camera.main.ScreenPointToRay(Input.mousePosition);//从摄像机发出到点击坐标的射线  
            RaycastHit hit;
            if (Physics.Raycast(ray, out hit)) {
                
                // 在随机选择场景中的一个Firework;
                int id = Random.Range(0, firework.Length);
                // 利用该Firework发射一个粒子,即烟花;
                firework[id].Emit(1);
                // 获取场景中属于firework[id]的运行中的粒子;
                int count = firework[id].GetParticles(m_Particles);
                // 根据鼠标点击位置计算烟花的飞行时间;
                float life_time = (hit.point.y - gameObject.transform.position.y) / firework[id].startSpeed;
                // 根据鼠标点击位置和烟花飞行时间计算烟花的水平速度;
                float x_v = (hit.point.x - gameObject.transform.position.x)/life_time;
                // 设置刚刚发射的粒子的飞行时间;
                m_Particles[count - 1].lifetime = life_time;
                // 设置刚刚发射的粒子的速度;
                m_Particles[count - 1].velocity = new Vector3(x_v, m_Particles[count - 1].velocity.y, m_Particles[count - 1].velocity.z);
                // 将改动应用到场景中;
                firework[id].SetParticles(m_Particles,count);
            }
        }
    }
}

我在场景中放置了四个Firework,每个Firework拥有不同的烟花颜色。每次发射烟花将从四个烟花发射器中选一个进行发射,并设定烟花的飞行时间以及飞行速度,保证烟花到达鼠标点击的位置就爆炸。

以下是最终的效果展示:

最终效果演示

项目地址

项目连接-传送门

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/qq_43996906/article/details/110478794

智能推荐

攻防世界_难度8_happy_puzzle_攻防世界困难模式攻略图文-程序员宅基地

文章浏览阅读645次。这个肯定是末尾的IDAT了,因为IDAT必须要满了才会开始一下个IDAT,这个明显就是末尾的IDAT了。,对应下面的create_head()代码。,对应下面的create_tail()代码。不要考虑爆破,我已经试了一下,太多情况了。题目来源:UNCTF。_攻防世界困难模式攻略图文

达梦数据库的导出(备份)、导入_达梦数据库导入导出-程序员宅基地

文章浏览阅读2.9k次,点赞3次,收藏10次。偶尔会用到,记录、分享。1. 数据库导出1.1 切换到dmdba用户su - dmdba1.2 进入达梦数据库安装路径的bin目录,执行导库操作  导出语句:./dexp cwy_init/[email protected]:5236 file=cwy_init.dmp log=cwy_init_exp.log 注释:   cwy_init/init_123..._达梦数据库导入导出

js引入kindeditor富文本编辑器的使用_kindeditor.js-程序员宅基地

文章浏览阅读1.9k次。1. 在官网上下载KindEditor文件,可以删掉不需要要到的jsp,asp,asp.net和php文件夹。接着把文件夹放到项目文件目录下。2. 修改html文件,在页面引入js文件:<script type="text/javascript" src="./kindeditor/kindeditor-all.js"></script><script type="text/javascript" src="./kindeditor/lang/zh-CN.js"_kindeditor.js

STM32学习过程记录11——基于STM32G431CBU6硬件SPI+DMA的高效WS2812B控制方法-程序员宅基地

文章浏览阅读2.3k次,点赞6次,收藏14次。SPI的详情简介不必赘述。假设我们通过SPI发送0xAA,我们的数据线就会变为10101010,通过修改不同的内容,即可修改SPI中0和1的持续时间。比如0xF0即为前半周期为高电平,后半周期为低电平的状态。在SPI的通信模式中,CPHA配置会影响该实验,下图展示了不同采样位置的SPI时序图[1]。CPOL = 0,CPHA = 1:CLK空闲状态 = 低电平,数据在下降沿采样,并在上升沿移出CPOL = 0,CPHA = 0:CLK空闲状态 = 低电平,数据在上升沿采样,并在下降沿移出。_stm32g431cbu6

计算机网络-数据链路层_接收方收到链路层数据后,使用crc检验后,余数为0,说明链路层的传输时可靠传输-程序员宅基地

文章浏览阅读1.2k次,点赞2次,收藏8次。数据链路层习题自测问题1.数据链路(即逻辑链路)与链路(即物理链路)有何区别?“电路接通了”与”数据链路接通了”的区别何在?2.数据链路层中的链路控制包括哪些功能?试讨论数据链路层做成可靠的链路层有哪些优点和缺点。3.网络适配器的作用是什么?网络适配器工作在哪一层?4.数据链路层的三个基本问题(帧定界、透明传输和差错检测)为什么都必须加以解决?5.如果在数据链路层不进行帧定界,会发生什么问题?6.PPP协议的主要特点是什么?为什么PPP不使用帧的编号?PPP适用于什么情况?为什么PPP协议不_接收方收到链路层数据后,使用crc检验后,余数为0,说明链路层的传输时可靠传输

软件测试工程师移民加拿大_无证移民,未受过软件工程师的教育(第1部分)-程序员宅基地

文章浏览阅读587次。软件测试工程师移民加拿大 无证移民,未受过软件工程师的教育(第1部分) (Undocumented Immigrant With No Education to Software Engineer(Part 1))Before I start, I want you to please bear with me on the way I write, I have very little gen...

随便推点

Thinkpad X250 secure boot failed 启动失败问题解决_安装完系统提示secureboot failure-程序员宅基地

文章浏览阅读304次。Thinkpad X250笔记本电脑,装的是FreeBSD,进入BIOS修改虚拟化配置(其后可能是误设置了安全开机),保存退出后系统无法启动,显示:secure boot failed ,把自己惊出一身冷汗,因为这台笔记本刚好还没开始做备份.....根据错误提示,到bios里面去找相关配置,在Security里面找到了Secure Boot选项,发现果然被设置为Enabled,将其修改为Disabled ,再开机,终于正常启动了。_安装完系统提示secureboot failure

C++如何做字符串分割(5种方法)_c++ 字符串分割-程序员宅基地

文章浏览阅读10w+次,点赞93次,收藏352次。1、用strtok函数进行字符串分割原型: char *strtok(char *str, const char *delim);功能:分解字符串为一组字符串。参数说明:str为要分解的字符串,delim为分隔符字符串。返回值:从str开头开始的一个个被分割的串。当没有被分割的串时则返回NULL。其它:strtok函数线程不安全,可以使用strtok_r替代。示例://借助strtok实现split#include <string.h>#include <stdio.h&_c++ 字符串分割

2013第四届蓝桥杯 C/C++本科A组 真题答案解析_2013年第四届c a组蓝桥杯省赛真题解答-程序员宅基地

文章浏览阅读2.3k次。1 .高斯日记 大数学家高斯有个好习惯:无论如何都要记日记。他的日记有个与众不同的地方,他从不注明年月日,而是用一个整数代替,比如:4210后来人们知道,那个整数就是日期,它表示那一天是高斯出生后的第几天。这或许也是个好习惯,它时时刻刻提醒着主人:日子又过去一天,还有多少时光可以用于浪费呢?高斯出生于:1777年4月30日。在高斯发现的一个重要定理的日记_2013年第四届c a组蓝桥杯省赛真题解答

基于供需算法优化的核极限学习机(KELM)分类算法-程序员宅基地

文章浏览阅读851次,点赞17次,收藏22次。摘要:本文利用供需算法对核极限学习机(KELM)进行优化,并用于分类。

metasploitable2渗透测试_metasploitable2怎么进入-程序员宅基地

文章浏览阅读1.1k次。一、系统弱密码登录1、在kali上执行命令行telnet 192.168.26.1292、Login和password都输入msfadmin3、登录成功,进入系统4、测试如下:二、MySQL弱密码登录:1、在kali上执行mysql –h 192.168.26.129 –u root2、登录成功,进入MySQL系统3、测试效果:三、PostgreSQL弱密码登录1、在Kali上执行psql -h 192.168.26.129 –U post..._metasploitable2怎么进入

Python学习之路:从入门到精通的指南_python人工智能开发从入门到精通pdf-程序员宅基地

文章浏览阅读257次。本文将为初学者提供Python学习的详细指南,从Python的历史、基础语法和数据类型到面向对象编程、模块和库的使用。通过本文,您将能够掌握Python编程的核心概念,为今后的编程学习和实践打下坚实基础。_python人工智能开发从入门到精通pdf