Unity3D手游开发日记(9) - 互动草的效果

所谓互动草,就是角色跑动或者释放技能,能影响草的摆动方向和幅度.

前面的文章早已经实现了风吹草动的效果,迟迟没有在Unity上面做互动草,是因为以前我在端游项目做过一套太过于牛逼的方案.在CE3的互动草的基础上扩展,效果好,但技术太复杂,效率开销也特别高. 如果在手机上,就得做一套简单高效的.

实现效果:从任意方向碰一下草,草就应该来回晃动,晃动幅度逐渐减小.多次触碰,效果应该叠加.这样的话就比较真实.

实现原理:用正玄波实现草来回摆动的简谐运动,用指数衰减来模拟阻力

实现步骤:

1.每个草挂一个脚本,来处理力的效果叠加

[csharp] view plain copy

  1. public class Force
  2. {
  3. public float m_Time = 0;
  4. public Vector4 m_Force;
  5. public Force(Vector4 force)
  6. {
  7. m_Force = force;
  8. }
  9. }
  10. public class GrassForce : MonoBehaviour
  11. {
  12. public List<Force> m_ForceList = null;
  13. public float m_WaveFrequency = 6.0f;
  14. public float m_Resistance = 0.25f;
  15. public float m_MaxForceMagnitude = 6.0f;
  16. public float m_AddForceTimeInterval = 0.5f;
  17. public int m_MaxForceNum = 3;
  18. private float m_LastAddTime = 0;
  19. private Material material;
  20. void Start()
  21. {
  22. material = gameObject.renderer.material;
  23. }
  24. void Update()
  25. {
  26. UpdateForce();
  27. }
  28. void OnBecameVisible()
  29. {
  30. enabled = true;
  31. }
  32. void OnBecameInvisible()
  33. {
  34. enabled = false;
  35. }
  36. public void AddForce(Vector3 force)
  37. {
  38. if (Time.time - m_LastAddTime > m_AddForceTimeInterval)
  39. {
  40. m_LastAddTime = Time.time;
  41. if (m_ForceList == null)
  42. m_ForceList = new List<Force>();
  43. if (m_ForceList.Count < m_MaxForceNum)
  44. {
  45. Vector4 newForce = new Vector4(force.x, 0, force.z, 0);
  46. if (newForce.magnitude > m_MaxForceMagnitude)
  47. newForce = newForce.normalized * m_MaxForceMagnitude;
  48. m_ForceList.Add(new Force(newForce));
  49. }
  50. }
  51. }
  52. private void UpdateForce()
  53. {
  54. if (m_ForceList == null)
  55. return;
  56. Vector4 accForce = Vector4.zero;
  57. for (int i = m_ForceList.Count - 1; i >= 0; --i)
  58. {
  59. if (m_ForceList[i].m_Force.magnitude > 0.1f)
  60. {
  61. // [-1, 1] 正玄波模拟简谐运动
  62. float wave_factor = Mathf.Sin(m_ForceList[i].m_Time * m_WaveFrequency);
  63. // 力的指数衰减
  64. float resistance_factor = easeOutExpo(1, 0, m_Resistance * Time.deltaTime);
  65. m_ForceList[i].m_Force *= resistance_factor;
  66. m_ForceList[i].m_Time += Time.deltaTime;
  67. // 累加
  68. accForce += m_ForceList[i].m_Force * wave_factor;
  69. }
  70. else
  71. {
  72. m_ForceList.RemoveAt(i);
  73. }
  74. }
  75. if (accForce != Vector4.zero)
  76. {
  77. if (material.HasProperty("_Force"))
  78. {
  79. accForce = transform.InverseTransformVector(accForce); // 世界空间转换到模型本地空间
  80. material.SetVector("_Force", accForce);
  81. }
  82. }
  83. }
  84. public float easeOutExpo(float start, float end, float value)
  85. {
  86. end -= start;
  87. return end * (-Mathf.Pow(2, -10 * value) + 1) + start;
  88. }
  89. }

2.如何确定哪些草受到影响,以及受力的方向?

[csharp] view plain copy

  1. public static void AddForceToGrass(int forceId, Transform transform)
  2. {
  3. ForceTable force = ForceTableMgr.Instance.GetDataById(forceId);
  4. if (force != null)
  5. {
  6. Vector3 relativeCenter = new Vector3(force.RelativeCenterX, force.RelativeCenterY, force.RelativeCenterZ);
  7. Vector3 center = transform.TransformPoint(relativeCenter);
  8. //Vector3 size = new Vector3(force.Length, force.Height, force.Width);
  9. Vector3 size = new Vector3(force.Width, force.Height, force.Length);
  10. // 方向矩阵
  11. Matrix4x4 m44 = Matrix4x4.TRS(Vector3.zero, Quaternion.Inverse(transform.rotation), Vector3.one);
  12. PhysicsUtil.AddForceToGrass((RangeType)force.RangeType, (ForceDirType)force.DirType, force.Strength, center, size, transform.forward, m44, force.Degree);
  13. }
  14. }
  15. private static void AddForceToGrass(RangeType type, ForceDirType dirType, float strength, Vector3 center, Vector3 size, Vector3 direction, Matrix4x4 m44, float degree = 360.0f)
  16. {
  17. if (type == RangeType.Sphere)
  18. {
  19. AddForceInSector(dirType, strength, center, size.x, direction, degree);
  20. }
  21. else if (type == RangeType.Cude)
  22. {
  23. AddForceInCube(dirType, strength, center, size, direction, m44, degree);
  24. }
  25. }

草可以看成一个点,计算和下面范围的相交.

1.圆形和扇形范围

圆形范围计算特别简单,计算距离即可.扇形范围只需要在圆形基础上再计算一次夹角即可,部分核心代码:

[csharp] view plain copy

  1. Vector3 dir = script.transform.position - center;
  2. if (dir.sqrMagnitude <= radius * radius)
  3. {
  4. dir.y = 0;
  5. if (Mathf.Abs(Vector3.Angle(dir, direction)) <= degree)
  6. {
  7. float factor = 0.25f + Mathf.Clamp01((radius - dir.magnitude) / radius) * 0.75f; // 衰减因子
  8. Vector3 forceDir;
  9. if (dirType == ForceDirType.ToTarget)
  10. forceDir = dir.normalized;
  11. else
  12. forceDir = -dir.normalized;
  13. script.AddForce(forceDir * factor * strength);
  14. }
  15. }

2.矩形范围

点和任意方向的矩形的计算,这个比较难.Unity本身也没提供此类相交API.不过熟悉引擎开发的应该知道AABB和OBB吧.其实矩形范围计算,就是计算点和OBB的相交.

点和AABB的相交计算很简单,因为AABB每条边都是和坐标轴平行或者垂直的.而OBB有方向,其实只需要把点矩阵变换到OBB所在的空间,就可以用AABB的方法来计算了.

[csharp] view plain copy

  1. public struct AABB
  2. {
  3. public Vector3 min;
  4. public Vector3 max;
  5. public AABB(Vector3 vmin, Vector3 vmax)
  6. {
  7. min = vmin;
  8. max = vmax;
  9. }
  10. }
  11. public struct OBB
  12. {
  13. public Matrix4x4 m44;
  14. public Vector3 h;               // half-length-vector
  15. public Vector3 c;               // center of obb
  16. public OBB(Matrix4x4 mat44, Vector3 hlv, Vector3 center)
  17. {
  18. m44 = mat44;
  19. h = hlv;
  20. c = center;
  21. }
  22. public OBB(Matrix4x4 mat44, AABB aabb)
  23. {
  24. m44 = mat44;
  25. h = (aabb.max - aabb.min) * 0.5f;
  26. c = (aabb.max + aabb.min) * 0.5f;
  27. }
  28. }
  29. // 点和AABB的相交
  30. public static bool Overlap_Point_AABB(Vector3 p, AABB aabb)
  31. {
  32. return ((p.x >= aabb.min.x && p.x <= aabb.max.x) && (p.y >= aabb.min.y && p.y <= aabb.max.y) && (p.z >= aabb.min.z && p.z <= aabb.max.z));
  33. }
  34. // 点和OBB的相交
  35. public static bool Overlap_Point_OBB(Vector3 p, Vector3 obbWorldPos, OBB obb)
  36. {
  37. AABB aabb = new AABB(obb.c - obb.h, obb.c + obb.h);
  38. Vector3 local_p = p - obbWorldPos;
  39. Vector3 t = obb.m44.MultiplyVector(local_p);
  40. return Overlap_Point_AABB(t, aabb);
  41. }

记住,OBB参数设置,中心一定要在世界原点,这样才方便计算

[csharp] view plain copy

  1. // 包围盒中心为世界原点.便于计算.
  2. Vector3 min = - size * 0.5f;
  3. Vector3 max = size * 0.5f;
  4. AABB aabb = new AABB(min, max);
  5. OBB obb = new OBB(m44, aabb);

相交和计算力方向:

[csharp] view plain copy

  1. if (PhysicsUtil.Overlap_Point_OBB(script.transform.position, center, obb))
  2. {
  3. // 暂时只支持Left_Right
  4. if (dirType == ForceDirType.Left_Right)
  5. {
  6. Vector3 dir = script.transform.position - center;
  7. dir = m44.MultiplyVector(dir);
  8. dir = (dir.x < 0) ? m44.transpose.MultiplyVector(Vector3.left) : m44.transpose.MultiplyVector(Vector3.right);
  9. Vector3 force = dir.normalized * strength;
  10. script.AddForce(force);
  11. }
  12. }

效果图:

1.圆形范围,力的方向从圆心到目标,模拟气浪把把草震开.

2.矩形范围,力的方向是玩家面向的左和右.模拟剑气把草劈开的感觉.

效率优化:

1.控制互动草的数量,这种草不能合批,谨记.

2.脚本加上了OnBecameVisible,OnBecameInvisible 只让摄像机内草起作用.

[csharp] view plain copy

时间: 2024-10-23 13:41:04

Unity3D手游开发日记(9) - 互动草的效果的相关文章

Unity3D手游开发日记(8) - 运动残影效果

2D游戏的残影很简单,美术做序列帧图片就行了,那么3D游戏的残影美术做不了,得靠程序员动态创建模型来处理. 实现原理也很简单: 1.间隔一定时间创建一个残影模型 [csharp] view plain copy GameObject go = GameObject.Instantiate(origin, pos, dir) as GameObject; 2.对残影模型采用特殊的shader,要简单高效 [csharp] view plain copy public class MotionGho

Unity3D手游开发日记(6) - 适合移动平台的水深处理

市面上大部分的手机游戏,水面都比较粗糙,也基本没发现谁做过水深的处理. 水深的处理在PC平台比较容易,因为很容易获得每个像素的深度,比如G-Buffer,有了像素的深度,就能计算出每个像素到水面的距离,实现水深alpha渐变. 但是在移动平台,又是万恶的浮点纹理...导致此方案不行. 但是方案都是人想出来的,我想了两种适合移动平台的方案 方案1:用水面顶点颜色保存alpha值来做水深渐变. 这种方案,要求水面的模型面片是格子的,就像地形网格一样,格子越密,alpha的精度才越高. 方案2:用贴图

Unity3D手游开发日记(4) - 适合移动平台的热浪扭曲

热浪扭曲效果的实现,分两部分,一是抓图,二是扭曲扰动.其中难点在于抓图的处理,网上的解决方案有两种,在移动平台都有很多问题,只好自己实现了一种新的方案,效果还不错. 网上方案1. 用GrabPass抓图 GrabPass在有的手机上是不支持的...效率也是问题,所以... 代码可以看看: [csharp] view plain copy Shader "Luoyinan/Distortion/HeatDistortion" { Properties { _NoiseTex ("

Unity3D手游开发日记(5) - 适合移动平台的植被随风摆动

一直在思考怎么让场景更有生机,我觉得植被的随风摆动是必不可少的.CE3引擎的植被bending就做得特别棒.我也准备在手机上做一套. 先分析一下植被摆动常见的几种做法.其实不管哪种做法,核心就是让植被顶点做动画,有的顶点动的少(比如树根),有的顶点动的多(比如树顶),根据怎么样的权重来动? 方案1:  用UV来做权重. 这种方案对UV展开有要求,要从0到1,只适合面片草,这样的话草的根部和顶部的摆动权重就是一个0到1的线性的变化,随便用一个正玄波就能实现简单摆动了, [csharp] view

Unity3D手游开发日记(1) - 移动平台实时阴影方案

阴影这个东西,说来就话长了,很多年前人们就开始研究出各种阴影技术,但都存在各种瑕疵和问题,直到近几年出现了PSSM,也就是CE3的CSM,阴影技术才算有个比较完美的解决方案.Unity自带的实时阴影,也是这种技术,在电脑上很成熟的.我也是目睹了阴影技术的发展,以前也自己写过一些阴影,本来以为以后再也不用担心阴影了,结果做移动游戏以后,发现情况没那么乐观.这种基于深度的shadow map 技术,需要浮点纹理的支持,但有的移动平台不支持浮点纹理,即使支持,这种技术开销也特别大.所以我们需要重新思考

Unity3D手游开发日记(7) - 适合移动平台的天气效果

腾讯的天涯明月刀的天气很棒,以前我也在CE3的基础上做了一个效果差不多的,但是在手机上,还是斜下固定视角的游戏,是否还需要一个天气系统? 而且没有G-Buffer的支持,很多牛逼效果实现不了,比如角色身上的湿润感,水面的波纹涟漪等 考虑了一下,觉得可以做一个简单高效的. 以下雨天为例子.一阵风吹来,天色逐渐变暗,突然下起了雨来,雨越来越大,夹杂着雷声和闪电,过几分钟,雨逐渐消失... 1.风 前面文章已经实现了风引起植被的摆动 2.天色 这个比较麻烦,由于移动平台的场景用的基本都是lightma

Unity3D手游开发日记(2) - 技能系统架构设计

我想把技能做的比较牛逼,所以项目一开始我就在思考,是否需要一个灵活自由的技能系统架构设计,传统的技能设计,做法都是填excel表,技能需要什么,都填表里,很死板,比如有的技能只需要1个特效,有的要10个,那么表格也得预留10个特效的字段.在代码里面也是写死一些东西,要增加和修改,就得改核心代码,如果我要把核心部分做成库封装起来,就很麻烦了. 能不能做成数据驱动的方式呢? 改技能文件就行了,即使要增加功能,也只需要扩展外部代码,而不用改核心代码, 我是这么来抽象一个技能的,技能由一堆触发器组成,比

Unity3D手游开发日记(3) - 场景加载进度条的完美方案

我以为做个进度条很简单,分分钟解决,结果折腾了一天才搞定,Unity有很多坑,要做完美需要逐一解决. 问题1:最简单的方法不能实现100%的进度 用最简单的方法来实现,不能实现100%的进度,原因是Unity加载完新场景立马就激活新场景了,无法显示最后的进度.解决办法就是使用allowSceneActivation来控制进入场景的时机. 问题2:使用allowSceneActivation后进度卡在90% 这个问题官网论坛也有人讨论,解决办法就是自己手动修补最后的10%, 问题3:进度条一顿一顿

Unity3D手游开发实践

<腾讯桌球:客户端总结> 本次分享总结,起源于腾讯桌球项目,但是不仅仅限于项目本身.虽然基于Unity3D,很多东西同样适用于Cocos.本文从以下10大点进行阐述: 架构设计 原生插件/平台交互 版本与补丁 用脚本,还是不用?这是一个问题 资源管理 性能优化 异常与Crash 适配与兼容 调试及开发工具 项目运营 ? 1.架构设计 好的架构利用大规模项目的多人团队开发和代码管理,也利用查找错误和后期维护. 框架的选择:需要根据团队.项目来进行选择,没有最好的框架,只有最合适的框架. 框架的使