Unity3D手游开发日记(9) - 互动草的效果
所谓互动草,就是角色跑动或者释放技能,能影响草的摆动方向和幅度.
前面的文章早已经实现了风吹草动的效果,迟迟没有在Unity上面做互动草,是因为以前我在端游项目做过一套太过于牛逼的方案.在CE3的互动草的基础上扩展,效果好,但技术太复杂,效率开销也特别高. 如果在手机上,就得做一套简单高效的.
实现效果:从任意方向碰一下草,草就应该来回晃动,晃动幅度逐渐减小.多次触碰,效果应该叠加.这样的话就比较真实.
实现原理:用正玄波实现草来回摆动的简谐运动,用指数衰减来模拟阻力
实现步骤:
1.每个草挂一个脚本,来处理力的效果叠加
- public class Force
- {
- public float m_Time = 0;
- public Vector4 m_Force;
- public Force(Vector4 force)
- {
- m_Force = force;
- }
- }
- public class GrassForce : MonoBehaviour
- {
- public List<Force> m_ForceList = null;
- public float m_WaveFrequency = 6.0f;
- public float m_Resistance = 0.25f;
- public float m_MaxForceMagnitude = 6.0f;
- public float m_AddForceTimeInterval = 0.5f;
- public int m_MaxForceNum = 3;
- private float m_LastAddTime = 0;
- private Material material;
- void Start()
- {
- material = gameObject.renderer.material;
- }
- void Update()
- {
- UpdateForce();
- }
- void OnBecameVisible()
- {
- enabled = true;
- }
- void OnBecameInvisible()
- {
- enabled = false;
- }
- public void AddForce(Vector3 force)
- {
- if (Time.time - m_LastAddTime > m_AddForceTimeInterval)
- {
- m_LastAddTime = Time.time;
- if (m_ForceList == null)
- m_ForceList = new List<Force>();
- if (m_ForceList.Count < m_MaxForceNum)
- {
- Vector4 newForce = new Vector4(force.x, 0, force.z, 0);
- if (newForce.magnitude > m_MaxForceMagnitude)
- newForce = newForce.normalized * m_MaxForceMagnitude;
- m_ForceList.Add(new Force(newForce));
- }
- }
- }
- private void UpdateForce()
- {
- if (m_ForceList == null)
- return;
- Vector4 accForce = Vector4.zero;
- for (int i = m_ForceList.Count - 1; i >= 0; --i)
- {
- if (m_ForceList[i].m_Force.magnitude > 0.1f)
- {
- // [-1, 1] 正玄波模拟简谐运动
- float wave_factor = Mathf.Sin(m_ForceList[i].m_Time * m_WaveFrequency);
- // 力的指数衰减
- float resistance_factor = easeOutExpo(1, 0, m_Resistance * Time.deltaTime);
- m_ForceList[i].m_Force *= resistance_factor;
- m_ForceList[i].m_Time += Time.deltaTime;
- // 累加
- accForce += m_ForceList[i].m_Force * wave_factor;
- }
- else
- {
- m_ForceList.RemoveAt(i);
- }
- }
- if (accForce != Vector4.zero)
- {
- if (material.HasProperty("_Force"))
- {
- accForce = transform.InverseTransformVector(accForce); // 世界空间转换到模型本地空间
- material.SetVector("_Force", accForce);
- }
- }
- }
- public float easeOutExpo(float start, float end, float value)
- {
- end -= start;
- return end * (-Mathf.Pow(2, -10 * value) + 1) + start;
- }
- }
2.如何确定哪些草受到影响,以及受力的方向?
- public static void AddForceToGrass(int forceId, Transform transform)
- {
- ForceTable force = ForceTableMgr.Instance.GetDataById(forceId);
- if (force != null)
- {
- Vector3 relativeCenter = new Vector3(force.RelativeCenterX, force.RelativeCenterY, force.RelativeCenterZ);
- Vector3 center = transform.TransformPoint(relativeCenter);
- //Vector3 size = new Vector3(force.Length, force.Height, force.Width);
- Vector3 size = new Vector3(force.Width, force.Height, force.Length);
- // 方向矩阵
- Matrix4x4 m44 = Matrix4x4.TRS(Vector3.zero, Quaternion.Inverse(transform.rotation), Vector3.one);
- PhysicsUtil.AddForceToGrass((RangeType)force.RangeType, (ForceDirType)force.DirType, force.Strength, center, size, transform.forward, m44, force.Degree);
- }
- }
- private static void AddForceToGrass(RangeType type, ForceDirType dirType, float strength, Vector3 center, Vector3 size, Vector3 direction, Matrix4x4 m44, float degree = 360.0f)
- {
- if (type == RangeType.Sphere)
- {
- AddForceInSector(dirType, strength, center, size.x, direction, degree);
- }
- else if (type == RangeType.Cude)
- {
- AddForceInCube(dirType, strength, center, size, direction, m44, degree);
- }
- }
草可以看成一个点,计算和下面范围的相交.
1.圆形和扇形范围
圆形范围计算特别简单,计算距离即可.扇形范围只需要在圆形基础上再计算一次夹角即可,部分核心代码:
- Vector3 dir = script.transform.position - center;
- if (dir.sqrMagnitude <= radius * radius)
- {
- dir.y = 0;
- if (Mathf.Abs(Vector3.Angle(dir, direction)) <= degree)
- {
- float factor = 0.25f + Mathf.Clamp01((radius - dir.magnitude) / radius) * 0.75f; // 衰减因子
- Vector3 forceDir;
- if (dirType == ForceDirType.ToTarget)
- forceDir = dir.normalized;
- else
- forceDir = -dir.normalized;
- script.AddForce(forceDir * factor * strength);
- }
- }
2.矩形范围
点和任意方向的矩形的计算,这个比较难.Unity本身也没提供此类相交API.不过熟悉引擎开发的应该知道AABB和OBB吧.其实矩形范围计算,就是计算点和OBB的相交.
点和AABB的相交计算很简单,因为AABB每条边都是和坐标轴平行或者垂直的.而OBB有方向,其实只需要把点矩阵变换到OBB所在的空间,就可以用AABB的方法来计算了.
- public struct AABB
- {
- public Vector3 min;
- public Vector3 max;
- public AABB(Vector3 vmin, Vector3 vmax)
- {
- min = vmin;
- max = vmax;
- }
- }
- public struct OBB
- {
- public Matrix4x4 m44;
- public Vector3 h; // half-length-vector
- public Vector3 c; // center of obb
- public OBB(Matrix4x4 mat44, Vector3 hlv, Vector3 center)
- {
- m44 = mat44;
- h = hlv;
- c = center;
- }
- public OBB(Matrix4x4 mat44, AABB aabb)
- {
- m44 = mat44;
- h = (aabb.max - aabb.min) * 0.5f;
- c = (aabb.max + aabb.min) * 0.5f;
- }
- }
- // 点和AABB的相交
- public static bool Overlap_Point_AABB(Vector3 p, AABB aabb)
- {
- 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));
- }
- // 点和OBB的相交
- public static bool Overlap_Point_OBB(Vector3 p, Vector3 obbWorldPos, OBB obb)
- {
- AABB aabb = new AABB(obb.c - obb.h, obb.c + obb.h);
- Vector3 local_p = p - obbWorldPos;
- Vector3 t = obb.m44.MultiplyVector(local_p);
- return Overlap_Point_AABB(t, aabb);
- }
记住,OBB参数设置,中心一定要在世界原点,这样才方便计算
- // 包围盒中心为世界原点.便于计算.
- Vector3 min = - size * 0.5f;
- Vector3 max = size * 0.5f;
- AABB aabb = new AABB(min, max);
- OBB obb = new OBB(m44, aabb);
相交和计算力方向:
- if (PhysicsUtil.Overlap_Point_OBB(script.transform.position, center, obb))
- {
- // 暂时只支持Left_Right
- if (dirType == ForceDirType.Left_Right)
- {
- Vector3 dir = script.transform.position - center;
- dir = m44.MultiplyVector(dir);
- dir = (dir.x < 0) ? m44.transpose.MultiplyVector(Vector3.left) : m44.transpose.MultiplyVector(Vector3.right);
- Vector3 force = dir.normalized * strength;
- script.AddForce(force);
- }
- }
效果图:
1.圆形范围,力的方向从圆心到目标,模拟气浪把把草震开.
2.矩形范围,力的方向是玩家面向的左和右.模拟剑气把草劈开的感觉.
效率优化:
1.控制互动草的数量,这种草不能合批,谨记.
2.脚本加上了OnBecameVisible,OnBecameInvisible 只让摄像机内草起作用.
Unity3D手游开发日记(9) - 互动草的效果的更多相关文章
- Unity3D手游开发日记(8) - 运动残影效果
2D游戏的残影很简单,美术做序列帧图片就行了,那么3D游戏的残影美术做不了,得靠程序员动态创建模型来处理. 实现原理也很简单: 1.间隔一定时间创建一个残影模型 GameObject go = Gam ...
- Unity3D手游开发日记(6) - 适合移动平台的水深处理
市面上大部分的手机游戏,水面都比较粗糙,也基本没发现谁做过水深的处理. 水深的处理在PC平台比较容易,因为很容易获得每个像素的深度,比如G-Buffer,有了像素的深度,就能计算出每个像素到水面的距离 ...
- Unity3D手游开发日记(4) - 适合移动平台的热浪扭曲
热浪扭曲效果的实现,分两部分,一是抓图,二是扭曲扰动.其中难点在于抓图的处理,网上的解决方案有两种,在移动平台都有很多问题,只好自己实现了一种新的方案,效果还不错. 网上方案1. 用GrabPass抓 ...
- Unity3D手游开发日记(5) - 适合移动平台的植被随风摆动
一直在思考怎么让场景更有生机,我觉得植被的随风摆动是必不可少的.CE3引擎的植被bending就做得特别棒.我也准备在手机上做一套. 先分析一下植被摆动常见的几种做法.其实不管哪种做法,核心就是让植被 ...
- Unity3D手游开发日记(7) - 适合移动平台的天气效果
腾讯的天涯明月刀的天气很棒,以前我也在CE3的基础上做了一个效果差不多的,但是在手机上,还是斜下固定视角的游戏,是否还需要一个天气系统? 而且没有G-Buffer的支持,很多牛逼效果实现不了,比如角色 ...
- Unity3D手游开发日记(2) - 技能系统架构设计
我想把技能做的比较牛逼,所以项目一开始我就在思考,是否需要一个灵活自由的技能系统架构设计,传统的技能设计,做法都是填excel表,技能需要什么,都填表里,很死板,比如有的技能只需要1个特效,有的要10 ...
- Unity3D手游开发日记(3) - 场景加载进度条的完美方案
我以为做个进度条很简单,分分钟解决,结果折腾了一天才搞定,Unity有很多坑,要做完美需要逐一解决. 问题1:最简单的方法不能实现100%的进度 用最简单的方法来实现,不能实现100%的进度,原因是U ...
- Unity3D手游开发日记(1) - 移动平台实时阴影方案
阴影这个东西,说来就话长了,很多年前人们就开始研究出各种阴影技术,但都存在各种瑕疵和问题,直到近几年出现了PSSM,也就是CE3的CSM,阴影技术才算有个比较完美的解决方案.Unity自带的实时阴影, ...
- Unity3D手游开发实践
<腾讯桌球:客户端总结> 本次分享总结,起源于腾讯桌球项目,但是不仅仅限于项目本身.虽然基于Unity3D,很多东西同样适用于Cocos.本文从以下10大点进行阐述: 架构设计 原生插件/ ...
随机推荐
- SpringMVC学习系列-后记 开启项目的OpenSessionInView
在系列的 SpringMVC学习系列(12) 完结篇 的示例项目中,由于当时考虑到OpenSessionInView会对性能有一定的影响,所以就没有配置项目的OpenSessionInView.在ma ...
- angular+bootstrap+MVC 之三,分页控件初级版
今天实现一个分页控件,效果如下: 1.HTML: <!doctype html> <!--suppress ALL --> <html ng-app="appT ...
- js call()和apply()
一.call()和apply(),实例如下: function add(a,b) { alert(a+b); } function sub(a,b) { alert(a-b); } a ...
- Android总结篇系列:Activity Intent Flags及Task相关属性
同上文一样,本文主要引用自网上现有博文,并加上一些自己的理解,在此感谢原作者. 原文地址: http://blog.csdn.net/liuhe688/article/details/6761337 ...
- Discuz有关问题解决办法汇总
1.Can not write to cache files, please check directory ./data/ and ./data/cache/ . 无法写入缓存 解决办法:在sour ...
- 织梦如何重新安装 织梦dir 二次安装
一般安装过一次织梦就不需要重新再去下载安装第二次了,我们可以在原有的基础上重新安装一次.这个是织梦cms后台自带的功能,还比较方便,下面小编就分享下如何重装织梦. 如果在本地重装要打开集成环境,在浏览 ...
- 对SSH三大框架的理解
SSH框架一般指的是Struts.Spring.Hibernate,后来Struts2代替了Struts.最近5年,Struts2已经被Spring MVC代替,而Hibernate基本也被iBati ...
- Spark Streaming源码解读之Receiver生成全生命周期彻底研究和思考
本期内容 : Receiver启动的方式设想 Receiver启动源码彻底分析 多个输入源输入启动,Receiver启动失败,只要我们的集群存在就希望Receiver启动成功,运行过程中基于每个Tea ...
- jQuery基础 -- 如何判断页面元素存在与否
在传统的Javascript里,当我们对某个页面元素进行某种操作前,最好先判断这个元素是否存在.原因是对一个不存在的元素进行操作是不允许的.例如: document.getElementById(&q ...
- CPU阿甘
本系列文章全部摘选自"码农翻身"公众号,仅供个人学习和分享之用.文章会给出原文的链接地址,希望不会涉及到版权问题. 个人感言:真正的知识是深入浅出的,码农翻身" 公共号将 ...