unity 模拟抛物线

简介: 具体的步骤:了解使用水平方向的速度是:v1=v0×cosθ竖直方向的速度是:v2=v0×sinθ-gty=v0×t-(gt^2)/2克隆线段,坐标信息,线段的长度,线段的间距。检测碰撞的点、控制线段的显示和隐藏、绘制弧克隆子弹,计算子弹的发射速度,子弹的朝向点击空格,初始化子弹数据,子弹发射在plyer下创建一个空物体为Arc;在代码中用创建一个空物体再添加组件LineRenderer,同时把这些空物体存到list中。

具体的步骤:

了解使用
水平方向的速度是:v1=v0×cosθ
竖直方向的速度是:v2=v0×sinθ-gt
y=v0×t-(gt^2)/2

克隆线段,坐标信息,线段的长度,线段的间距。
检测碰撞的点、控制线段的显示和隐藏、绘制弧
克隆子弹,计算子弹的发射速度,子弹的朝向
点击空格,初始化子弹数据,子弹发射

在plyer下创建一个空物体为Arc;在代码中用创建一个空物体再添加组件LineRenderer,同时把这些空物体存到list中。

segmentCount 线段的个数
LineRendererList 为LineRenderer类型
segmentwidth 设置的是0.01f

public void CreateArc()
{
for (int i = 0; i < segmentCount; i++)
{
GameObject cl = new GameObject("Arc_" + i);
cl.transform.SetParent(player.transform.Find("Arc"));
LineRendererList.Add(cl.AddComponent());

LineRendererList[i].startWidth = segmentwidth;
LineRendererList[i].endWidth = segmentwidth;
LineRendererList[i].material = LineMaterial;
LineRendererList[i].shadowCastingMode = ShadowCastingMode.Off;
LineRendererList[i].receiveShadows = false;
LineRendererList[i].lightProbeUsage = LightProbeUsage.Off;
LineRendererList[i].reflectionProbeUsage = ReflectionProbeUsage.Off;
LineRendererList[i].enabled = false;
}
}
获取在每个时间时,弧线的位置

IsUseGravity 是bool类型
使用公式:y=v0×t-(gt^2)/2
v0是我们手动定义的速度,
public Vector3 GetTimePosition(float time)
{
Vector3 usegravity = IsUseGravity ? Physics.gravity : Vector3.zero;
Vector3 result = player.transform.position+time arcSpeed + (0.5f time time) usegravity;
return result;
}
设置线段的坐标

public void SetPostion(int index, Vector3 StartPostion, Vector3 endPostion)
{
LineRendererList[index].enabled = true;
LineRendererList[index].SetPosition(0, StartPostion);
LineRendererList[index].SetPosition(1, endPostion);
}
每个线段的长度,和线段之间的间距

float timeStep = arcDuration / segmentCount;
[Range(0, 1)] public float segmentBreak;
当物体碰撞时返回碰撞的时间

HitLayer是地板的当前层级
Physics.Linecast在API有解释,在两点之间绘制射线,在这线段之间如有

碰撞返回为真

碰撞时间:如a到b绘制射线,a为第一个点,b为第二个点。
设置a点的时间是A,设置b点的时间是B。
timeStep 为A与B的时间差
AB 之间的距离为C,
碰撞点与A之间的距离 为D
碰撞的点一定是在线段之间,所以时间=(D/C)*timeStep +A

public float GetHitTime(out RaycastHit hitInfo)
{
float timeStep = arcDuration / segmentCount;
float startTime_1 = 0.0f;
Vector3 startPos_1 = GetTimePosition(startTime_1);
hitInfo = new RaycastHit();
for (int i = 0; i < segmentCount; i++)
{
float endtime_1 = timeStep + startTime_1;
Vector3 endpos_1 = GetTimePosition(endtime_1);

if (Physics.Linecast(startPos_1, endpos_1, out hitInfo, HitLayer))
{
Bullseye.transform.position = hitInfo.point+Vector3.up*0.1F;
Bullseye.SetActive(true);
float distance = Vector3.Distance(startPos_1, endpos_1);
float hitdistance = hitInfo.distance;
return startTime_1 + (hitdistance / distance) * timeStep;
}
startPos_1 = endpos_1;
startTime_1 = endtime_1;
}
return float.MaxValue;
}
绘制弧线

在unity中抛物线的速度是距离,我们需要初速度=(目标的方向*速度)
timeStep 设置一个线段的时间,segmentBreak线段之间的间距时间

private void DrawArc()
{
float timeStep = arcDuration / segmentCount;
arcSpeed = transform.forward * startVelocity;
currentTimeOffset = (Time.time - arcTimeOffset) *0.2f;
if (currentTimeOffset > (timeStep + segmentBreak))
{
currentTimeOffset = 0;
arcTimeOffset = Time.time;
}
float starttime = currentTimeOffset;

hitTime = GetHitTime(out hitInfo);
int i = 0;
for (i = 0; i < segmentCount; i++)
{
float endTime = starttime + timeStep;
Vector3 StartPos = GetTimePosition(starttime);
Vector3 Endpos = GetTimePosition(endTime);
SetPostion(i, StartPos, Endpos);
starttime += timeStep + segmentBreak;
if (starttime > hitTime)
{
break;
}
}
HideLine(i);
}
克隆子弹

bullet = GameObject.CreatePrimitive(PrimitiveType.Capsule);
子弹的发射速度

velocityX 水平的速度
velocityY 垂直速度
tanθ=对边/临边

public void BulletMove()
{
float time = Time.time - currentTime;
Vector3 playerXZ = transform.position;
playerXZ.y = 0;
velocityX = arcSpeed.magnitude;
velocityY =startVelocity Mathf.Sin(launchAngle Mathf.Deg2Rad) + Physics.gravity.y * time ;
Vector3 speed =transform.forward;
speed *= velocityX;
speed.y = velocityY;

bullet.transform.position += (speed) * Time.fixedDeltaTime;
Vector3 speedXZ = speed;
speedXZ.y = 0;

float tempTan = speedXZ.magnitude / speed.y;
float hu = Mathf.Atan(tempTan);
float angle = hu * Mathf.Rad2Deg;
bullet.transform.eulerAngles = new Vector3(angle, bullet.transform.eulerAngles.y, bullet.transform.eulerAngles.z);

if (Vector3.Distance(bullet.transform.position, Bullseye.transform.position) < 0.1f)
{
IsMove = !IsMove;
}
}
目的:
当玩家移动,绘制的弧线也可以移动,子弹防止朝向不对
public void InitBullet()
{
bullet.transform.localScale = Vector3.one * 0.3f;
bullet.transform.position = player.transform.position;
bullet.transform.rotation = player.transform.rotation;
}
点击空格时:
要算出当前时间
currentTime = Time.time;
初始化子弹数据
考虑当旋转角度>180时
if (launchAngle > 180)
{
launchAngle = 360 - launchAngle;
}

IsMove = !IsMove

ControlMove();
DrawArc();
放在FixedUpdate()
更多unity2018的功能介绍请到paws3d爪爪学院查找。

相关文章
|
5月前
|
流计算 Python
Python实现动态银河系:模拟旋转的银河动画
Python实现动态银河系:模拟旋转的银河动画
71 0
|
7月前
|
数据可视化 数据处理 Python
使用Python模拟绘制自由落体运动过程中的抛物线
使用Python模拟绘制自由落体运动过程中的抛物线
135 0
|
数据可视化 图形学
Unity射线编程详解
Unity射线编程详解
107 0
【Unity3D--自由观察模型】模型自动旋转+触屏旋转和缩放
展示3D模型,同时实现模型自动旋转和触屏旋转和缩放
434 0
如何做一个俄罗斯方块5:形状碰撞检测(下)
其实,两侧的碰撞判断跟我们上一节讲过的向下移动的碰撞判断原理是一样的,向下碰撞检测的是每一个方块下方的位置是否有其它方块,那么向左/右碰撞检测的就是每个方块左/右侧的位置是否有其他的方块。
323 0
|
开发工具
如何做一个俄罗斯方块4:形状碰撞检测(上)
在游戏开发中,我们所说的“碰撞”经常指的是物理碰撞,什么是物理碰撞呢?一般的在游戏开发工具中都会包含一个叫做“物理引擎”的东西,它的作用就是在游戏中模拟出现实中的物理效果。例如,我们扔一个东西,这个东西会因为重力而下落,最终落到地上,与地面发生碰撞。在游戏中,我们可以借助物理引擎,来模拟出东西下落掉到地面上的效果。当东西掉到地面上时,我们就说这个东西与地面发生了碰撞。
317 0
Qt Qwdget 汽车仪表知识点拆解7 图像绘制,旋转
先贴上效果图,注意,没有写逻辑,都是乱动的
126 0
Qt Qwdget 汽车仪表知识点拆解7 图像绘制,旋转
|
图形学
Unity在UI界面上显示3D模型/物体,控制模型旋转
Unity3D物体在UI界面的显示 本文提供全流程,中文翻译。 Chinar 坚持将简单的生活方式,带给世人!(拥有更好的阅读体验 —— 高分辨率用户请根据需求调整网页缩放比例) Chinar —...
5165 0
|
图形学
Unity 3D中的射线与碰撞检测
创建一条射线Ray需要指明射线的起点(origin)和射线的方向(direction)。这两个参数也是Ray的成员变量。注意,射线的方向在设置时如果未单位化,Unity 3D会自动进行单位归一化处理。
4446 0