Unity 双指旋转

思路都是:

每一帧以两指坐标生成一个向量Vector2,不同帧比较向量的角度变化Vector2.Angle()。

简化版:参考EasyTouch的实现

public class TouchTest : MonoBehaviour
{

	Touch oldTouch1;  //上次触摸点1(手指1)
	Touch oldTouch2;  //上次触摸点2(手指2)

	void Update()
	{
		if (Input.touchCount <= 1) {
			return;
		}

		Touch touch1 = Input.GetTouch(0);
		Touch touch2 = Input.GetTouch(1);

		//启用双指,尚未旋转
		if (touch2.phase == TouchPhase.Began) {
			oldTouch2 = touch2;
			oldTouch1 = touch1;
			return;
		}
		if (touch1.phase == TouchPhase.Moved || touch2.phase == TouchPhase.Moved) {
			Vector2 curVec = touch2.position - touch1.position;
			Vector2 oldVec = oldTouch2.position - oldTouch1.position;
			float angle = Vector2.Angle(oldVec, curVec);
			angle *= Mathf.Sign(Vector3.Cross(oldVec, curVec).z);

			transform.Rotate(new Vector3(0, 0, angle));

			oldTouch1 = touch1;
			oldTouch2 = touch2;
		}
	}
}



另一种写法,参考的博客已失效,这里贴一下

const float pinchTurnRatio = Mathf.PI / 2;
	//用这个数值来判定旋转的最小角度,自己决定大小
	const float minTurnAngle = 0;

	/// <summary>
	///   The delta of the angle between two touch points
	/// </summary>
	static public float turnAngleDelta;
	/// <summary>
	///   The angle between two touch points
	/// </summary>
	static public float turnAngle;

	/// <summary>
	///   Calculates Pinch and Turn - This should be used inside LateUpdate
	/// </summary>
	static public void Calculate()
	{
		//pinchDistance = pinchDistanceDelta = 0;
		turnAngle = turnAngleDelta = 0;

		// if two fingers are touching the screen at the same time ...
		if (Input.touchCount == 2) {
			Touch touch1 = Input.touches[0];
			Touch touch2 = Input.touches[1];

			// ... if at least one of them moved ...如果有一根手指有移动
			if (touch1.phase == TouchPhase.Moved || touch2.phase == TouchPhase.Moved) {
				// ... or check the delta angle between them ...
				//当前两手指与X轴的角
				turnAngle = Angle(touch1.position, touch2.position);
				//之前两手指与X的角度
				float prevTurn = Angle(touch1.position - touch1.deltaPosition,
									   touch2.position - touch2.deltaPosition);
				//旋转的偏移量, 增量角
				turnAngleDelta = Mathf.DeltaAngle(prevTurn, turnAngle);

				// ... if it's greater than a minimum threshold, it's a turn!
				//增量角如果大于设定值
				if (Mathf.Abs(turnAngleDelta) > minTurnAngle) {
					turnAngleDelta *= pinchTurnRatio;
				} else {
					turnAngle = turnAngleDelta = 0;
				}
			}
		}

		//鼠标判断事件,这个只是个测试
#if UNITY_EDITOR
		MouseRotateEvent();
#endif
	}

	/// <summary>
	/// 取两个向量之间的角度 与X轴
	/// </summary>
	/// <param name="pos1"></param>
	/// <param name="pos2"></param>
	/// <returns></returns>
	static private float Angle(Vector2 pos1, Vector2 pos2)
	{
		//取两个向量相减
		Vector2 from = pos2 - pos1;

		//X轴
		Vector2 to = new Vector2(1, 0);

		//与X轴夹角
		float result = Vector2.Angle(from, to);

		//向量的叉乘
		Vector3 cross = Vector3.Cross(from, to);
		//如果超出360度
		if (cross.z > 0) {
			//用360来减这个角度
			result = 360f - result;
		}

		return result;
	}

#if UNITY_EDITOR
	static bool isPress = false;
	static float preAngle = 0;

	/// <summary>
	/// 鼠标右键旋转事件,并非完美实现,因为没有判断角度为正还是为负。
	/// </summary>
	static public void MouseRotateEvent()
	{
		if (Input.GetMouseButton(1)) {
			//第一次按下
			if (!isPress) {
				//上一帧的角度
				preAngle = Angle(Input.mousePosition, Vector2.zero);
				isPress = true;
			} else {
				//转动的角度
				turnAngle = Angle(Input.mousePosition, Vector2.zero);
				//旋转的偏移量, 增量角
				turnAngleDelta = Mathf.DeltaAngle(preAngle, turnAngle);
				Log.log("增量角:"+turnAngleDelta);

				//增量角如果大于设定值
				if (Mathf.Abs(turnAngleDelta) > minTurnAngle) {
					turnAngleDelta *= pinchTurnRatio;
				} else {
					turnAngle = turnAngleDelta = 0;
				}

				//更新为这一帧
				preAngle = Angle(Input.mousePosition, Vector2.zero);
			}
		} else {
			isPress = false;
		}
	}
#endif






	/// <summary>
	/// 绕点旋转,或者是Z轴旋转?
	/// </summary>
	public bool isRotateAround = false;

	//目标
	public Transform target;

	//围绕点
	protected Vector3 axis {
		get {
			return new Vector3(target.position.x, transform.position.y, target.position.z);
		}
	}

	//void Start()
	//{
	//	if (target == null) {
	//		Debug.Log("target is null!");
	//		enabled = false;
	//	}
	//}

	void Update()
	{
		Calculate();

		if (Mathf.Abs(turnAngleDelta) > 0) {
			if (isRotateAround) {
				//围点旋转
				transform.RotateAround(axis, Vector3.up, -turnAngleDelta);
				//目视对方
				transform.LookAt(target);
			} else {
				Quaternion desiredRotation = transform.rotation;
				Vector3 rotationDeg = Vector3.zero;
				//z轴旋转
				rotationDeg.z = -turnAngleDelta;
				desiredRotation *= Quaternion.Euler(rotationDeg);
				transform.rotation = desiredRotation;
			}
		}
	}




  • 5
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要实现在Unity中使用双指放大/缩小图片,可以使用Unity的Input类检测多点触控事件,例如TouchPhase.Moved和TouchPhase.Ended事件。具体实现步骤如下: 1. 创建一个Image对象,并将其添加到场景中。 2. 在脚本中使用Input类检测多点触控事件。 3. 获取触控点之间的距离,并根据距离的变化来缩放图片。 下面是一个示例代码: ``` using UnityEngine; using UnityEngine.UI; public class ZoomImage : MonoBehaviour { public Image image; private Vector2[] touchPoints = new Vector2[2]; private float prevDistance = 0f; void Update() { // 检测多点触控事件 if (Input.touchCount == 2) { Touch touch1 = Input.GetTouch(0); Touch touch2 = Input.GetTouch(1); // 记录触控点的位置 touchPoints[0] = touch1.position; touchPoints[1] = touch2.position; if (touch1.phase == TouchPhase.Moved || touch2.phase == TouchPhase.Moved) { // 计算触控点之间的距离 float distance = Vector2.Distance(touchPoints[0], touchPoints[1]); if (prevDistance == 0f) { prevDistance = distance; } else { // 计算距离的变化并缩放图片 float scaleFactor = distance / prevDistance; image.transform.localScale *= scaleFactor; prevDistance = distance; } } } else if (Input.touchCount == 0) { // 重置距离 prevDistance = 0f; } } } ``` 在该脚本中,我们首先检测是否有两个触控点,然后记录它们的位置。在触控点移动时,我们计算它们之间的距离,并根据距离的变化来缩放图片。当没有触控点时,我们重置距离,以便下一次缩放操作。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值