zoukankan      html  css  js  c++  java
  • Unity3D手游开发日记(9)

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

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

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

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

    实现步骤:

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

    [csharp] view plain copy
     
     在CODE上查看代码片派生到我的代码片
    1.     public class Force  
    2.     {  
    3.         public float m_Time = 0;  
    4.         public Vector4 m_Force;  
    5.   
    6.         public Force(Vector4 force)  
    7.         {  
    8.             m_Force = force;  
    9.         }  
    10.     }  
    11.   
    12.     public class GrassForce : MonoBehaviour  
    13.     {  
    14.         public List<Force> m_ForceList = null;  
    15.         public float m_WaveFrequency = 6.0f;  
    16.         public float m_Resistance = 0.25f;  
    17.         public float m_MaxForceMagnitude = 6.0f;  
    18.         public float m_AddForceTimeInterval = 0.5f;  
    19.         public int m_MaxForceNum = 3;  
    20.   
    21.         private float m_LastAddTime = 0;  
    22.         private Material material;  
    23.   
    24.         void Start()  
    25.         {  
    26.             material = gameObject.renderer.material;  
    27.         }  
    28.   
    29.         void Update()  
    30.         {  
    31.             UpdateForce();  
    32.         }  
    33.   
    34.         void OnBecameVisible()  
    35.         {  
    36.             enabled = true;  
    37.         }  
    38.         void OnBecameInvisible()  
    39.         {  
    40.             enabled = false;  
    41.         }  
    42.   
    43.         public void AddForce(Vector3 force)  
    44.         {  
    45.             if (Time.time - m_LastAddTime > m_AddForceTimeInterval)  
    46.             {  
    47.                 m_LastAddTime = Time.time;  
    48.   
    49.                 if (m_ForceList == null)  
    50.                     m_ForceList = new List<Force>();  
    51.   
    52.                 if (m_ForceList.Count < m_MaxForceNum)  
    53.                 {  
    54.                     Vector4 newForce = new Vector4(force.x, 0, force.z, 0);  
    55.                     if (newForce.magnitude > m_MaxForceMagnitude)  
    56.                         newForce = newForce.normalized * m_MaxForceMagnitude;  
    57.   
    58.                     m_ForceList.Add(new Force(newForce));  
    59.                 }    
    60.             }               
    61.         }  
    62.   
    63.         private void UpdateForce()  
    64.         {  
    65.             if (m_ForceList == null)  
    66.                 return;  
    67.   
    68.             Vector4 accForce = Vector4.zero;  
    69.             for (int i = m_ForceList.Count - 1; i >= 0; --i)  
    70.             {  
    71.                 if (m_ForceList[i].m_Force.magnitude > 0.1f)  
    72.                 {  
    73.                     // [-1, 1] 正玄波模拟简谐运动  
    74.                     float wave_factor = Mathf.Sin(m_ForceList[i].m_Time * m_WaveFrequency);  
    75.   
    76.                     // 力的指数衰减      
    77.                     float resistance_factor = easeOutExpo(1, 0, m_Resistance * Time.deltaTime);  
    78.                     m_ForceList[i].m_Force *= resistance_factor;  
    79.   
    80.                     m_ForceList[i].m_Time += Time.deltaTime;  
    81.   
    82.                     // 累加  
    83.                     accForce += m_ForceList[i].m_Force * wave_factor;  
    84.                 }  
    85.                 else  
    86.                 {  
    87.                     m_ForceList.RemoveAt(i);  
    88.                 }  
    89.             }  
    90.   
    91.             if (accForce != Vector4.zero)  
    92.             {  
    93.                 if (material.HasProperty("_Force"))  
    94.                 {  
    95.                     accForce = transform.InverseTransformVector(accForce); // 世界空间转换到模型本地空间  
    96.                     material.SetVector("_Force", accForce);  
    97.                 }    
    98.             }  
    99.         }  
    100.   
    101.         public float easeOutExpo(float start, float end, float value)  
    102.         {  
    103.             end -= start;  
    104.             return end * (-Mathf.Pow(2, -10 * value) + 1) + start;  
    105.         }  
    106.     }  
    2.如何确定哪些草受到影响,以及受力的方向?
    [csharp] view plain copy
     
     在CODE上查看代码片派生到我的代码片
    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.         // 方向矩阵  
    12.         Matrix4x4 m44 = Matrix4x4.TRS(Vector3.zero, Quaternion.Inverse(transform.rotation), Vector3.one);   
    13.   
    14.         PhysicsUtil.AddForceToGrass((RangeType)force.RangeType, (ForceDirType)force.DirType, force.Strength, center, size, transform.forward, m44, force.Degree);  
    15.     }  
    16. }  
    17.   
    18. private static void AddForceToGrass(RangeType type, ForceDirType dirType, float strength, Vector3 center, Vector3 size, Vector3 direction, Matrix4x4 m44, float degree = 360.0f)  
    19. {  
    20.     if (type == RangeType.Sphere)  
    21.     {  
    22.         AddForceInSector(dirType, strength, center, size.x, direction, degree);  
    23.     }  
    24.     else if (type == RangeType.Cude)  
    25.     {  
    26.         AddForceInCube(dirType, strength, center, size, direction, m44, degree);  
    27.     }  
    28. }  

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

    1.圆形和扇形范围

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

    [csharp] view plain copy
     
     在CODE上查看代码片派生到我的代码片
    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.   
    9.                             Vector3 forceDir;  
    10.                             if (dirType == ForceDirType.ToTarget)  
    11.                                 forceDir = dir.normalized;  
    12.                             else  
    13.                                 forceDir = -dir.normalized;  
    14.   
    15.                             script.AddForce(forceDir * factor * strength);  
    16.                         }  
    17.                     }  

    2.矩形范围

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

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

    [csharp] view plain copy
     
     在CODE上查看代码片派生到我的代码片
    1. public struct AABB  
    2. {  
    3.     public Vector3 min;  
    4.     public Vector3 max;  
    5.   
    6.     public AABB(Vector3 vmin, Vector3 vmax)  
    7.     {  
    8.         min = vmin;  
    9.         max = vmax;  
    10.     }  
    11. }  
    12.   
    13. public struct OBB  
    14. {  
    15.     public Matrix4x4 m44;             
    16.     public Vector3 h;               // half-length-vector  
    17.     public Vector3 c;               // center of obb   
    18.   
    19.     public OBB(Matrix4x4 mat44, Vector3 hlv, Vector3 center)  
    20.     {  
    21.         m44 = mat44;  
    22.         h = hlv;  
    23.         c = center;  
    24.     }  
    25.   
    26.     public OBB(Matrix4x4 mat44, AABB aabb)  
    27.     {  
    28.         m44 = mat44;  
    29.         h = (aabb.max - aabb.min) * 0.5f;     
    30.         c = (aabb.max + aabb.min) * 0.5f;     
    31.     }  
    32. }  
    33.   
    34. // 点和AABB的相交  
    35. public static bool Overlap_Point_AABB(Vector3 p, AABB aabb)  
    36. {  
    37.     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));  
    38. }  
    39.   
    40. // 点和OBB的相交  
    41. public static bool Overlap_Point_OBB(Vector3 p, Vector3 obbWorldPos, OBB obb)  
    42. {  
    43.     AABB aabb = new AABB(obb.c - obb.h, obb.c + obb.h);  
    44.     Vector3 local_p = p - obbWorldPos;  
    45.     Vector3 t = obb.m44.MultiplyVector(local_p);   
    46.     return Overlap_Point_AABB(t, aabb);  
    47. }  
    记住,OBB参数设置,中心一定要在世界原点,这样才方便计算
    [csharp] view plain copy
     
     在CODE上查看代码片派生到我的代码片
    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
     
     在CODE上查看代码片派生到我的代码片
    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.   
    10.         Vector3 force = dir.normalized * strength;  
    11.         script.AddForce(force);  
    12.     }  
    13. }  


    效果图:

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

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

    效率优化:

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

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

    [csharp] view plain copy
     
    1.   

     
     
  • 相关阅读:
    Android:使用 DownloadManager 进行版本更新
    Android:UI 沉浸式体验,适合第一屏的引导图片、预览图片。
    Android:相机适配及图片处理的一些问题
    Android: 设置 app 字体大小不跟随系统字体调整而变化
    Android: TextView 及其子类通过代码和 XML 设置字体大小的存在差异的分析
    SQLMap 学习
    我的书单
    macos
    linux BufferedImage.createGraphics()卡住不动
    Linux 中ifconfig和ip addr命令看不到ip
  • 原文地址:https://www.cnblogs.com/czaoth/p/5785801.html
Copyright © 2011-2022 走看看