实现WSAD移动方向,鼠标右键旋转角度,滚轮实现缩放
在此之前,我们先讲清楚几个比较容易混淆的点:
1:Transform.forward是Transform组件的一个Vector3类型的属性,代表游戏对象在世界坐标系中的朝向,即“前方向”,它的值是(0,0,1),它表示了一个对象的X轴方向,一般用于计算游戏对象前进的方向,比如用于移动、旋转和射线检测等场景。
2:Transform.Right的数学表达式是(1, 0, 0)。它表示了一个对象的X轴方向,或者说是它的右侧方向
值得注意的是,它们都是单位向量,一般只用于方向计算
3:Input.GetAxis("Horizontal")和Input.GetAxis("Vertical"),input.GetAxis("Mouse X")
它们是分别获取水平方向输入和竖直方向输入的函数,这个函数会返回一个范围在-1到1之间的浮点数,表示当前水平方向上的输入状态
比如Input.GetAxis("Mouse X")
返回鼠标在水平方向上的移动,以浮点数表示。当鼠标向右移动时,返回正值;当鼠标向左移动时,返回负值。如果鼠标不动,返回0
Input.GetAxis("Mouse ScrollWheel"),向前滚动是正数,向后滚动时负数
4:欧拉角和四元数
在unity中的rotation就是使用了欧拉角,欧拉角指的是以三个轴为基准,通过绕不同轴的旋转角度来表示一个物体的旋转状态,通常有XYZ、ZYX、ZXY等多种旋转顺序。但是,欧拉角会存在万向锁问题,即在某些特定情况下,两个轴的旋转会发生重合,导致旋转计算异常
四元数则是通过四元数的定义和运算来实现旋转,四元数实际上是一个四元组,包含了一个实部和三个虚部。它们的定义方式和欧拉角不同,可以避免万向锁问题,并且旋转顺序不会影响旋转结果。四元数还可以通过插值运算实现平滑旋转,适用于需要连续旋转的情况。
我把代码分成了三块
第一个模块是WSAD控制前后左右,这段比较简单,直接定义一个V3类型的移动方向,再使用Translate即可
第二个模块是使用QE控制上下视角,定义一个变量upDown,每次按键时候改变1,然后直接把位置赋值到相机位置即可
第三个模块是使用鼠标右键进行旋转,设置两个旋转向量rotateX,rotateY,Y方向旋转限值minAngle和maxAngle,然后把向量和速度加到相机的欧拉角即可
有一个难点,在刚开始调试的时候,第一次点击鼠标右键进行旋转时,屏幕一直跳动,最后经过GPT的帮助,最终解决了问题!爱死GPT
using UnityEngine;
public class CameraController : MonoBehaviour
{//WSAD相机移动速度public float moveSpeed = 1.0f;//相机当前移动方向private Vector3 moveDirection;//QE相机的目标高度private float upDistance;public float upSpeed = 0.2f;//鼠标右键控制旋转private float rotateX,rotateY;public float sensitivity =0.5f;//控制鼠标在Y方向上的限值public float minAngle = -90f;public float maxAngle = 90f;//记录之前的欧拉角,避免跳屏private Vector3 currentRotation,lastPosition;//鼠标滚轮控制缩放public float zoomSpeed = 10f;private float zoomDistance = 0f;private void Start(){currentRotation = transform.eulerAngles;}void Update(){// 使用WSAD控制相机前后左右移动比较简单,直接获取当前的移动方向,然后使用Translate移动即可{moveDirection = new Vector3(Input.GetAxis("Horizontal"), 0, Input.GetAxis("Vertical"));transform.Translate(moveDirection * moveSpeed * Time.deltaTime);}//使用QE控制相机的上下,使用unDown参数进行单位变化{if (Input.GetKey(KeyCode.Q)){upDistance -= upSpeed * Time.deltaTime;transform.Translate(transform.up * upDistance, Space.World);}else if(Input.GetKeyUp(KeyCode.Q)){upDistance = 0;}if (Input.GetKey(KeyCode.E)){upDistance += upSpeed * Time.deltaTime;transform.Translate(transform.up * upDistance, Space.World);}else if(Input.GetKeyUp(KeyCode.E)){upDistance = 0;}}//使用鼠标右键来控制相机旋转//Mouse X,向右移动返回正值,Mouse Y,向上移动鼠标为正值{if (Input.GetMouseButtonDown(1)){// 记录当前的欧拉角//currentRotation = transform.eulerAngles;//记录鼠标位置lastPosition = Input.mousePosition;}if (Input.GetMouseButton(1)){//设置偏移量Vector3 offset = Input.mousePosition - lastPosition;rotateX += offset.x * sensitivity;rotateY -= offset.y * sensitivity;//给Y方向的旋转加上限值函数rotateY = Mathf.Clamp(rotateY, minAngle, maxAngle);//改变当前的欧拉角transform.eulerAngles = new Vector3(rotateY, rotateX, 0f);// 将保存的欧拉角重新赋值回去transform.eulerAngles += currentRotation;lastPosition = Input.mousePosition;}}//使用滚轮来控制物体的缩放{//获取滚轮的滚动幅度和方向zoomDistance += Input.GetAxis("Mouse ScrollWheel") * zoomSpeed;//使用限值函数来限定缩放的范围zoomDistance = Mathf.Clamp(zoomDistance, -10f, 10f);//最后定位transform.position = transform.position + transform.forward * Input.GetAxis("Mouse ScrollWheel") * zoomSpeed;}}
}
直接将代码复制,挂到Camera即可运行!