亲宝软件园·资讯

展开

Unity手势解锁 Unity实现移动端手势解锁功能

LLLLL__ 人气:0

一、效果演示

二、实现思路

——当鼠标选中一个密码按钮时开始记录输入的数字和鼠标的起始位置
——当鼠标按下过程中,始终根据记录的鼠标起始位置和当前鼠标的位置两个点绘制线段并添加到线段的列表中,并一直清空掉列表中除了最后一个线段外的其余线段
——当鼠标按下过程中,如果有覆盖到其他的密码按钮,则根据起始的密码按钮与当前的密码按钮两个点绘制线段并重新记录输入的数字和鼠标起始位置

三、实现过程

——创建9个密码块,并依次命名为1、2.....9,并设置tag为PasswordBlock

——编写生成LineRenderer的方法,初始化LineRenderer属性的方法,绘制线的方法以及清空线的方法
注意绘制线段时,需要将起始位置和结束位置的z轴置为0

——编写记录密码和删除密码的方法

四、完整代码(挂载到手势解锁界面的物体身上)

using UnityEngine;
using System.Collections.Generic;
using UnityEngine.EventSystems;
 
public class GestureUnlock : MonoBehaviour, IBeginDragHandler, IEndDragHandler, IDragHandler
{
 private List<GameObject> lrList = new List<GameObject>();//存储每个LineRenderer的列表
 private List<GameObject> passwordButtonList = new List<GameObject>();//存储每个密码按钮的列表
 
 private Vector3 startPos;//鼠标开始的位置
 
 //线的参数
 public Color startColor = Color.black;//线开始的颜色
 public Color endColor = Color.black;//线结束的颜色
 public float width = 0.1f;//线宽度
 public int vertices = 90;//顶点数
 
 public string password;//密码
 private string inputPassword;//输入的密码
 
 /// <summary>
 /// 刷新线段(拖拽过程中一直刷新)
 /// </summary>
 private void RefreshLine()
 {
 if (passwordButtonList.Count == 0)
 {
 return;
 }
 
 LineRenderer uncompleteLR = SpawnLineRenderer(false);
 InitLine(uncompleteLR);
 DrawLine(uncompleteLR, startPos, ScreenToWorld(Input.mousePosition));
 ClearLine(false);
 }
 
 /// <summary>
 /// 绘制已经连线完成的线段
 /// </summary>
 private void DrawCompleteLine(Vector3 endPos)
 {
 LineRenderer completeLR = SpawnLineRenderer(true);
 InitLine(completeLR);
 DrawLine(completeLR, startPos, endPos);
 }
 
 /// <summary>
 /// 记录密码
 /// </summary>
 /// <param name="_passwordBlock">密码块物体</param>
 private void RecordPassword(GameObject _passwordButton)
 {
 passwordButtonList.Add(_passwordButton);
 inputPassword += _passwordButton.name;
 
 startPos = _passwordButton.transform.position;//记录起始位置
 }
 
 /// <summary>
 /// 删除密码
 /// </summary>
 private void DeletePassword()
 {
 passwordButtonList.Clear();
 inputPassword = "";
 }
 
 public void OnBeginDrag(PointerEventData eventData)
 {
 ClearLine(true);//每次开始拖拽时清空所有线段
 
 GameObject go = eventData.pointerEnter;
 if (go != null && go.tag == "PasswordButton" && IsExistInPasswordBlockList(go) == false)
 {
 RecordPassword(go);//记录密码
 }
 }
 
 public void OnEndDrag(PointerEventData eventData)
 {
 ClearUnCompleteLine();//清除未完成的线段
 }
 
 public void OnDrag(PointerEventData eventData)
 {
 RefreshLine();//刷新线段(拖拽过程中一直刷新)
 
 GameObject go = eventData.pointerEnter;
 if (passwordButtonList.Count != 0 && go != null && go.tag == "PasswordButton" && IsExistInPasswordBlockList(go) == false)
 {
 DrawCompleteLine(go.transform.position);//绘制已经连线完成的线段
 
 RecordPassword(go);//记录密码
 }
 }
 
 #region 线段相关操作
 
 /// <summary>
 /// 生成LineRenderer
 /// </summary>
 private LineRenderer SpawnLineRenderer(bool isCompleteLine)
 {
 LineRenderer uncompleteLR = new GameObject().AddComponent<LineRenderer>();
 uncompleteLR.material = new Material(Shader.Find("Sprites/Default"));
 lrList.Add(uncompleteLR.gameObject);
 if (isCompleteLine)
 {
 uncompleteLR.gameObject.name = "CompleteLine";
 }
 else
 {
 uncompleteLR.gameObject.name = "UncompleteLine";
 }
 return uncompleteLR;
 }
 
 /// <summary>
 /// 初始化线
 /// </summary>
 private void InitLine(LineRenderer _uncompleteLR)
 {
 _uncompleteLR.startColor = startColor;
 _uncompleteLR.endColor = endColor;
 _uncompleteLR.startWidth = width;
 _uncompleteLR.endWidth = width;
 _uncompleteLR.numCapVertices = vertices;
 _uncompleteLR.numCornerVertices = vertices;
 }
 
 /// <summary>
 /// 两点绘制一条直线
 /// </summary>
 /// <param name="_uncompleteLR">线段</param>
 /// <param name="startPos">起始位置</param>
 /// <param name="endPos">结束位置</param>
 private void DrawLine(LineRenderer _uncompleteLR, Vector3 startPos, Vector3 endPos)
 {
 _uncompleteLR.positionCount = 2;
 startPos.z = 0;
 endPos.z = 0;
 _uncompleteLR.SetPosition(0, startPos);
 _uncompleteLR.SetPosition(1, endPos);
 }
 
 /// <summary>
 /// 清除线段
 /// </summary>
 /// <param name="clearAll">是否清除全部线段</param>
 private void ClearLine(bool clearAll)
 {
 if (lrList.Count == 0)
 {
 return;
 }
 
 for (int i = lrList.Count - 1; i >= 0; i--)
 {
 GameObject go = lrList[i];
 if (clearAll)
 {
 Destroy(go);
 lrList.Remove(go);
 }
 else
 {
 if (go.name != "CompleteLine" && i != lrList.Count - 1)
 {
  Destroy(go);
  lrList.Remove(go);
 }
 }
 }
 
 if (clearAll)
 {
 DeletePassword();
 }
 }
 
 /// <summary>
 /// 清除未完成的线段(每次拖拽结束时清除)
 /// </summary>
 private void ClearUnCompleteLine()
 {
 if (lrList.Count == 0)
 {
 return;
 }
 
 GameObject go = lrList[lrList.Count - 1];
 Destroy(go);
 lrList.Remove(go);
 }
 
 #endregion
 
 #region 工具方法
 
 /// <summary>
 /// 当前密码块是否存在于密码块列表中
 /// </summary>
 /// <param name="_passwordBlock">密码块</param>
 private bool IsExistInPasswordBlockList(GameObject _passwordButton)
 {
 if (passwordButtonList.Count == 0)
 {
 return false;
 }
 
 if (passwordButtonList.Contains(_passwordButton))
 {
 return true;
 }
 else
 {
 return false;
 }
 }
 
 /// <summary>
 /// 屏幕坐标转世界坐标
 /// </summary>
 /// <param name="screenPos">屏幕坐标位置</param>
 /// <param name="camera">相机</param>
 /// <returns>转换后的世界坐标</returns>
 private Vector3 ScreenToWorld(Vector3 screenPos, Camera camera = null)
 {
 if (camera == null)
 {
 camera = Camera.main;
 }
 Vector3 _screenPos = new Vector3(screenPos.x, screenPos.y, -camera.transform.position.z);
 Vector3 v = camera.ScreenToWorldPoint(_screenPos);
 return v;
 }
 
 #endregion
}

加载全部内容

相关教程
猜你喜欢
用户评论