标签:观察 结果 一个 意义 sep 射击 lse code 补充
我们在设计游戏的时候,经常会进行坐标系的变换,Unity为我们提供了多个变换的API,这里主要对它们的使用做一个总结整理!
在Unity中我们通常会用到以下几个坐标系下的点:
transform.position | transform.rotattion
获取得到的位置和旋转信息都是基于世界坐标系的,可以说,我们的很大一部分操作都是基于世界坐标系。(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
轴数据了!。标签:观察 结果 一个 意义 sep 射击 lse code 补充
原文地址:https://www.cnblogs.com/joemono/p/13186660.html