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. atitit.GMT UTC Catitit.GMT UTC CST DST CET 星期 月份 节日 时间的不同本质and起源

    atitit.GMT UTC Catitit.GMT UTC CST DST CET 星期 月份 节日 时间的不同本质and起源 1. GMT(Greenwich Mean Time)是格林尼治平时 ...

  2. HTML5游戏开发进阶指南(亚马逊5星畅销书,教你用HTML5和JavaScript构建游戏!)

    HTML5游戏开发进阶指南(亚马逊星畅销书,教你用HTML5和JavaScript构建游戏!) [印]香卡(Shankar,A.R.)著 谢光磊译 ISBN 978-7-121-21226-0 201 ...

  3. 解决 "Windows 无法启动Mongo DB服务 错误:1067 进程意外终止"

    在启动MongoDB服务时,有时会报上图所示的错误,解决方案为: 1. MongoDB安装目录\data\将此文件夹下的mongod.lock删除 2. 查看官方文档或按照上一篇安装文章检查是否设置d ...

  4. java生成竖排文字图片

    package com.kadang.designer.web.action;import java.awt.Color;import java.awt.Font;import java.awt.Fo ...

  5. android eclipse关联源码,以及源码(代码)以及jar查看软件

    1.eclipse关联源码 步骤一:自已百度下载相应版本的源码,我这里是4.2也就是API=17的版本. 步骤二:找到你安装sdk的目录, G:\Program Files\adt-bundle-wi ...

  6. android: SQLite查询数据

    掌握了查询数据的方法之后,你也就将数据库的 CRUD 操 作全部学完了.不过千万不要因此而放松,因为查询数据也是在 CRUD 中最复杂的一种 操作. 我们都知道 SQL 的全称是 Structured ...

  7. android:configChanges属性总结

    http://blog.csdn.net/zhaokaiqiang1992/article/details/19921703 android中的组件Activity在manifest.xml文件中可以 ...

  8. JPA的事务注解@Transactional使用总结

    在项目开发过程中,如果您的项目中使用了Spring的@Transactional注解,有时候会出现一些奇怪的问题,例如: 明明抛了异常却不回滚? 嵌套事务执行报错? ...等等 很多的问题都是没有全面 ...

  9. Android调用浏览器打开网址遇到的问题

    我自己的手机(一加一代,升级了氢OS),然后在点击游戏内一个"隐私政策"-- 需要打开一个网页,然后就crash了.出错的信息如下: 完全是看不出来,然后我单独写了一个demo来测 ...

  10. Javascript:常用函数封装

    //cookie function setCookie(name, value, iDay) { if(iDay!==false) { var oDate=new Date(); oDate.setD ...