标签:mod npoi 文章 例子 top DPoS height 理解 不同的
在Unity中我们通常会用到以下几个坐标系下的点:
transform.position | transform.rotation
获取得到的位置和旋转信息都是基于世界坐标系的,可以说,我们的很大一部分操作都是基于世界坐标系。(0,0)
,右上角为(1,1)
,我们在设计分屏游戏的时候可以通过设置摄像机所占据的视口空间来控制。(0,0)
,但右上角为(screen.width,screen.height)
。比如游戏的分辨率为500*600
,则screen.width=500;screen.height=600
Input.mousePosition
来获取鼠标的位置,这里获取到的鼠标位置是基于屏幕坐标的。通过该函数返回的是Vector3
类型的变量,但z
分量始终为0。//1.屏幕转世界坐标 Vector3 Camera.main.ScreenToWorldPoint(new Vector3(screenPos.x , screenPos.y , zInfo)); //2.世界转屏幕坐标 Vector3 Camera.main.WorldToScreenPoint(new Vector3(worldPos.x , worldPos.y , worldPos.z)); //3.世界转视口坐标 Vector3 Camera.main.WorldToViewportPoint(); //4.视口转世界坐标 Vector3 Camera.main.ViewportToWorldPoint(new Vector3(viewPortPos.x , viewPortPos.y , zInfo)); //5.视口转屏幕坐标 Vector3 Camera.main.ViewportToScreenPoint(); //6.屏幕转视口坐标 Vector3 Camera.main.ScreenToViewportPoint();
观察这些个函数,首先一个很明显的共同点,就是这些函数都是Camera
的成员函数,输入和输出都为Vector3
类型的变量。也即这些函数都是针对当前摄像机的一个变换操作。这很容易理解,因为3D游戏中的坐标从模型空间到最终的屏幕空间经过了model
,view
,projection
,以及之后的NDC变换
等,其中除model
是用于从模型空间到世界空间的变换外,之后的view
,projection
都是基于摄像机的。他们会随着使用相机的变化而变化。至于具体的内容,变化过程,这里不做过多描述,可以查看网上的相关文章。
当然,我们在使用这些API的时候,只需要清楚我们的输入和输出的内容及其关系就好了。
接下来,我们来聊一聊这些函数:
首先是屏幕坐标和世界坐标的相互转换:
WorldToScreenPoint函数接收一个世界空间下的位置信息,然后返回其所在的屏幕空间位置,以及其相对于摄像机的深度信息,该深度信息由世界空间下摄像机和输入位置的z值来决定。 一个例子是:摄像机的位置为(0.0, 0.0 , -10.0),输入的位置为(0.0,0.0,1.0)。则返回的结果为(screen.width/2 , screen.height/2 , 1-(-10)); 注意摄像机指向-z方向! ScreenToWorldPoint则是与之相反,输入屏幕空间位置以及相应的深度信息(注意深度信息应该为目标z值金和相机z值的差值),可以返回其所在的世界坐标位置。
视口坐标和世界坐标与之相似:
WorldToViewportPoint:输入世界坐标,返回的是对应的点所在的视口位置,当然以及其相对于摄像机的深度信息(距离)
ViewportToWorldPoint:输入视口坐标(记得对应的深度信息),返回点所在的世界坐标
视口坐标和屏幕坐标非常简单,正如上面的说明,只要知道分辨率就可以轻松转换,这里不再赘述。
我们在设计某些游戏的时候(比如摄像机固定不动的类型),会对物体的运动范围进行限制,以防止其跑出边界。比如一盒横屏的飞行射击游戏(雷电),我们可以获得物体的位置信息:transform.position
,我们希望对物体的x
,y
两个轴向的移动进行限制(同时冻结物体的z轴移动)。一个简单的思路如下:
public float leftBorder; public float rightBorder; public float topBorder; public float bottomBorder; .... Vector3 leftBtm_cornerPos = Camera.main.ViewportToWorldPoint(new Vector3(0f, 0f, Mathf.Abs(-Camera.main.transform.position.z))); //这里的z轴在正交视图下意义不大 Vector3 rightTop_cornerPos = Camera.main.ViewportToWorldPoint(new Vector3(1f, 1f, Mathf.Abs(-Camera.main.transform.position.z))); .... leftBorder = leftBtm_cornerPos.x; rightBorder = rightTop_cornerPos.x; topBorder = rightTop_cornerPos.y; bottomBorder = leftBtm_cornerPos.y; .... if (pos.x <= leftBorder) { pos.x = leftBorder; } else if (pos.x >= rightBorde { pos.x = rightBorder; } if (pos.y <= bottomBorder) { pos.y = bottomBorder; } else if (pos.y >= topBorder) { pos.y = topBorder; }
上面的代码通过ViewportToWorldPoint
获取到了四个边界。之后通过限制位置的x
、y
轴向移动就可以控制避免移动过度。
z
轴并无太大的作用。即使我使用ViewportToWorldPoint
的时候使用0
作为z
轴数据,也一样没关系。但是对于透视投影就不太一样了。若这里使用的透视投影,在不同的深度下,其的边界范围肯定也会变化,这个时候就必须输入正确的z
轴数据了。标签:mod npoi 文章 例子 top DPoS height 理解 不同的
原文地址:https://www.cnblogs.com/unity3ds/p/11301239.html