CSGO里的火焰效果和真实的情况比较像,能沿着遮挡物前进,如下是模仿效果。

  

  思路比较简单,开始想的是一圈一圈发出去,但是前圈与后圈的联系不好做,换种思路,每个方向发射一条线,这样根据上一个位置的方位先向前进,如果前面有遮挡,则计算好新的位置与方向,反之前面没有遮挡,选择合适的位置,并从这个方向的上面向下检测,检测这个方向的垂直位置有没遮挡,如果有遮挡,计算新的方向与位置,没有,则表明延展不下去。如下图所示.

  

  相关主要代码:  

    //根据一条线的上一个节点,确定这个节点如何定位
bool forward(ref Vector3 pos, ref Quaternion quat)
{
var forward = quat * Vector3.forward;
var up = quat * Vector3.up; var nextPos = pos + forward * radius;
RaycastHit hit;
//前面有没挡住
//Debug.DrawRay(nextPos, forward, Color.red, 100);
var hitForward = Physics.Raycast(nextPos, forward, out hit, radius * );
var hitDown = false;
//如果前面没有挡住,检查垂直位置
if (!hitForward)
{
//height用来控制垂直扩展的高度
var upPos = pos + * forward * radius + up * height * ;
//Debug.DrawRay(upPos, -up, Color.green, 100);
hitDown = Physics.Raycast(upPos, -up, out hit, height * );
}
if (hitForward || hitDown)
{
//新的位置
pos = hit.point + hit.normal * prefabHeight;
//Debug.DrawRay(pos, hit.normal, Color.blue, 100);
quat = Quaternion.FromToRotation(up, hit.normal) * quat;
return true;
}
return false;
}

forward

  如上代码每句都加上了注释,我们添加了一个radius参数,用来表示每次前进的步子大小,radius越小,则越密集,然后添加一个height参数用来表示垂直方向升降能力,可以看下图。

  确定一条线上显示后,我们只要考虑沿着周围全部扩展就行,在这我们用level表示扩展多少层,层数越多,我们每条线的相隔的角度也应该越小,如下效果。

  

  相关代码:  

    //分成多条线向外扩散
public void extend()
{
//分成多条线
count = (int)( * level * Mathf.PI);
//每条偏移角度
angle = 360.0f / count;
var up = transform.rotation * Vector3.up;
for (int i = ; i < count; i++)
{
var curAngle = angle * i;
var curQuat = Quaternion.AngleAxis(curAngle, up);
var quat = curQuat * transform.rotation;
//forward(transform.position, quat, level, i);
StartCoroutine(forward(transform.position, quat, level, i));
}
} //每条线向前扩展,自动翻越遮挡
public IEnumerator forward(Vector3 pos, Quaternion quat, int level, int place)
{
var npos = pos;
var nquat = quat;
int curLevel = ; var prePos = npos;
while (curLevel <= level && forward(ref npos, ref nquat))
{
//Debug.DrawLine(prePos, npos, Color.blue, 100f, true);
if (bCreate(curLevel, place))
{
var offset = offsetPos(nquat, );
var obj = Instantiate(getPrefab(curLevel), npos + offset, nquat);
//Debug.DrawLine(prePos, npos, Color.blue, 100f, true);
var duration = getDuration(curLevel);
Destroy(obj, duration);
}
yield return new WaitForSeconds(deltaTime);
curLevel++;
prePos = npos;
}
}

extend

  这些蓝线确定了火焰沿线步局,但是现在最中心会比较密集,我们需要有选择的是否生成,如下图:

  

  相关代码:  

    //确定当前层上的某个位置是否需要生成
bool bCreate(int level, int place)
{
//当前需要显示的个数
var levelCount = (int)( * level * Mathf.PI);
var levelLength = Mathf.RoundToInt((float)count / levelCount);
//每层起点偏移一点位置
var offset = place + level;
return offset % levelLength == ;
}

bCreate

  这样就能简单在每层生成合适的个数,其中最上面有个prefabHeight,用来控制模型中心与下边的高度,用来控制生成的模型贴着遮挡物,如图:

  

  如下是完整代码:  

using UnityEngine;
using System.Collections; public class FireMove : MonoBehaviour
{
public GameObject[] particlePrefabs;
[Tooltip("半径越小,模型越密集")]
public float radius = 0.5f;
[Tooltip("高度越大,模型能越过更高的遮挡物")]
public float height = 0.5f;
[Tooltip("层级越多,向外扩展的导数越多")]
public int level = ;
[Tooltip("每层间隔生成时间")]
public float deltaTime = 0.1f;
[Tooltip("控制火焰显示时间")]
public float duration = 5.0f;
[Tooltip("水平偏移")]
public float offsetHeight = 0.1f;
[Tooltip("高度偏移")]
public float offsetPanel = 0.2f;
[Tooltip("模型与地面的距离")]
public float prefabHeight = 0.01f; private int count;
private float angle; #if UNITY_EDITOR
public void Update()
{
if (Input.GetKeyDown(KeyCode.K))
{
extend();
}
}
#endif public void attack(int level)
{
this.level = level;
extend();
} //分成多条线向外扩散
public void extend()
{
//分成多条线
count = (int)( * level * Mathf.PI);
//每条偏移角度
angle = 360.0f / count;
var up = transform.rotation * Vector3.up;
for (int i = ; i < count; i++)
{
var curAngle = angle * i;
var curQuat = Quaternion.AngleAxis(curAngle, up);
var quat = curQuat * transform.rotation;
//forward(transform.position, quat, level, i);
StartCoroutine(forward(transform.position, quat, level, i));
}
} //每条线向前扩展,自动翻越遮挡
public IEnumerator forward(Vector3 pos, Quaternion quat, int level, int place)
{
var npos = pos;
var nquat = quat;
int curLevel = ; var prePos = npos;
while (curLevel <= level && forward(ref npos, ref nquat))
{
//Debug.DrawLine(prePos, npos, Color.blue, 100f, true);
if (bCreate(curLevel, place))
{
var offset = offsetPos(nquat, );
var obj = Instantiate(getPrefab(curLevel), npos + offset, nquat);
//Debug.DrawLine(prePos, npos, Color.blue, 100f, true);
var duration = getDuration(curLevel);
Destroy(obj, duration);
}
yield return new WaitForSeconds(deltaTime);
curLevel++;
prePos = npos;
}
} //确定当前层上的某个位置是否需要生成
bool bCreate(int level, int place)
{
//当前需要显示的个数
var levelCount = (int)( * level * Mathf.PI);
var levelLength = Mathf.RoundToInt((float)count / levelCount);
//每层起点偏移一点位置
var offset = place + level;
return offset % levelLength == ;
}
public GameObject getPrefab(int curLevel)
{
var prfabCount = particlePrefabs.Length;
//var t = Mathf.RoundToInt(prfabCount * (curLevel - 1) / level);
var index = Random.Range(, prfabCount);
return particlePrefabs[index];
} //加上偏移
public Vector3 offsetPos(Quaternion quat, int level)
{
var xAxis = quat * Vector3.right * Random.Range(-offsetPanel, offsetPanel) * level;
var zAxis = quat * Vector3.forward * Random.Range(-offsetPanel, offsetPanel) * level;
var yAxis = quat * Vector3.up * Random.Range(-offsetHeight, offsetHeight) * level;
return xAxis + zAxis + yAxis;
} //每层的生命周期
float getDuration(int level)
{
var levelDuration = duration - * level * deltaTime;
return levelDuration;
} //根据一条线的上一个节点,确定这个节点如何定位
bool forward(ref Vector3 pos, ref Quaternion quat)
{
var forward = quat * Vector3.forward;
var up = quat * Vector3.up; var nextPos = pos + forward * radius;
RaycastHit hit;
//前面有没挡住
//Debug.DrawRay(nextPos, forward, Color.red, 100);
var hitForward = Physics.Raycast(nextPos, forward, out hit, radius * );
var hitDown = false;
//如果前面没有挡住,检查垂直位置
if (!hitForward)
{
//height用来控制垂直扩展的高度
var upPos = pos + * forward * radius + up * height * ;
//Debug.DrawRay(upPos, -up, Color.green, 100);
hitDown = Physics.Raycast(upPos, -up, out hit, height * );
}
if (hitForward || hitDown)
{
//新的位置
pos = hit.point + hit.normal * prefabHeight;
//Debug.DrawRay(pos, hit.normal, Color.blue, 100);
quat = Quaternion.FromToRotation(up, hit.normal) * quat;
return true;
}
return false;
}
}

FireMove

  直接放到一个物体上就可以了,其中radius设为相应的火焰半径大小就可,如果要密一些,降低一些就行,相应offset参数用来设定水平与垂直上的偏移,免的给人太规律了,代码上都有详细注释,可以自己改动。

用Unity模仿CSGO里的火焰效果的更多相关文章

  1. GJM : 用Unity模仿CSGO里的火焰效果 [转载]

    感谢您的阅读.喜欢的.有用的就请大哥大嫂们高抬贵手"推荐一下"吧!你的精神支持是博主强大的写作动力以及转载收藏动力.欢迎转载! 版权声明:本文原创发表于 [请点击连接前往] ,未经 ...

  2. js模仿jquery里的几个方法parent, parentUntil, children

    有时工作需要, 也是接着上一章的方法, 用js模仿jquery里的几个方法parent, parentUntil, children. function parent(node){ return no ...

  3. unity, 在OnDisable里一定要将Cloth禁掉

    如果在OnDisable中不将Cloth组件禁掉,则当物体再次激活时布料将变形.

  4. js模仿jquery里的几个方法next, pre, nextAll, preAll

    /*siblings函数, 选取node的所有兄弟节点*/ function siblings(node){ if(node.nodeType === 1){ node.flag = true; // ...

  5. 模仿jq里的选择器和color样式

    (function(){ HTMLElement.prototype.css = function () { var option; if (arguments.length > 0) { op ...

  6. Unity 5 中的全局光照技术详解

    貌似是某位好人翻译的 https://unity3d.com/cn/learn/tutorials/topics/graphics/unity-5-lighting-and-rendering#rd? ...

  7. Unity 5 中的全局光照技术详解(建议收藏)

    本文整理自Unity全球官方网站,原文:UNITY 5 - LIGHTING AND RENDERING (文章较长,请耐心阅读)简介全局光照,简称GI,是一个用来模拟光的互动和反弹等复杂行为的算法, ...

  8. Unity 5--全局光照技术

    本文整理自Unity全球官方网站,原文:UNITY 5 - LIGHTING AND RENDERING 简介全局光照,简称GI,是一个用来模拟光的互动和反弹等复杂行为的算法,要精确的仿真全局光照非常 ...

  9. 04、Unity 5--全局光照技术

    本文整理自Unity全球官方网站,原文:UNITY 5 - LIGHTING AND RENDERING 简介全局光照,简称GI,是一个用来模拟光的互动和反弹等复杂行为的算法,要精确的仿真全局光照非常 ...

随机推荐

  1. PHP部分字符串函数汇总

    PHP部分字符串函数汇总 提交 我的评论 加载中 已评论 PHP部分字符串函数汇总 2015-03-10 PHP100中文网 PHP100中文网 PHP100中文网 微信号 功能介绍 互联网开发者社区 ...

  2. ajax异步请求Response.Redirect重定向

    一个ajax异步请求报错->捕获异常->重定向错误提示页面.  一个简单的流程 结果一直搞不定.重定向无效.各种百度之. 后来突然想起 ajax的请求是不能在后台重定向的. 如果硬要重定向 ...

  3. IFrame 高度自适应的两种方式 .

    iframe 高度自适应一般是指: iframe 本身的高度 =  内容高度. 这样做可以使最外层不出现滚动条. 如果网页内容使用了Ajax方式填充内容的话. 由于内容是动态的. 以上方式应该变为: ...

  4. ConcurrentDictionary 对决 Dictionary+Locking

    在 .NET 4.0 之前,如果我们需要在多线程环境下使用 Dictionary 类,除了自己实现线程同步来保证线程安全之外,我们没有其他选择. 很多开发人员肯定都实现过类似的线程安全方案,可能是通过 ...

  5. Hadoop HDFS 架构设计

    HDFS 简介 Hadoop Distributed File System,简称HDFS,是一个分布式文件系统. HDFS是高容错性的,可以部署在低成本的硬件之上,HDFS提供高吞吐量地对应用程序数 ...

  6. google hack 之 查询语法

    google hack 之 查询语法 文/玄魂   前言 谷歌网页搜索技术,大部分在百度等搜索引擎中也适用.同样,这些搜索技术是来源于传统数据库检索技术,因而,对这部分的学习,能为后续章节的数据库检索 ...

  7. [WinAPI] API 14 [获取、设置文件属性和时间]

    >_< 为了获取文件属性,用户可以使用GetFileAttributes与GetFileAttributesEx函数. GetFileAttributesEx函数除了返回文件属性外,还返回 ...

  8. Homebrew- MAC上的包管理利器

    包管理器是神马东西?让我们看看wikipedia上的介绍. In software, a package management system, also called package manager, ...

  9. LeadTools Android 入门教学——运行第一个Android Demo

    LeadTools 有很多Windows平台下的Demo,非常全面,但是目前开发手机应用的趋势也越来越明显,LeadTools也给大家提供了10个Android的Demo,这篇文章将会教你如何运行第一 ...

  10. DataGridView绑定复杂实体(属性本身又是实体)

    今天我们来讨论下一个实体中某个属性又是实体的类型的集合绑定到DataGridView上的问题. 先来写一个Student类 public class Student { public int Stud ...