Unity實現(xiàn)虛擬搖桿
更新時間:2020年04月14日 10:42:23 作者:LLLLL__
這篇文章主要為大家詳細(xì)介紹了Unity實現(xiàn)虛擬搖桿,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
本文實例為大家分享了Unity實現(xiàn)虛擬搖桿的具體代碼,供大家參考,具體內(nèi)容如下

面板上設(shè)置一些屬性,比如搖桿拖拽的距離,是否始終可視,是否限制虛擬搖桿位置(我是把虛擬搖桿限制在了屏幕的左下區(qū)域)。
使用GetDirAndLength()方法去獲得移動的方向和長度即可

using UnityEngine;
/// <summary>
/// 虛擬搖桿管理器
/// </summary>
public class VirtualJoystickManager : MonoBehaviour
{
private static VirtualJoystickManager _instance;
public static VirtualJoystickManager Instance
{
get
{
if (_instance == null)
{
_instance = FindObjectOfType<VirtualJoystickManager>();
}
return _instance;
}
}
[Header("是否始終可視")]
public bool alwaysVisible;//是否始終可視
[Header("是否限制虛擬搖桿位置")]
public bool restrictVirtualJoystickPos;//是否限制虛擬搖桿位置
[Header("虛擬搖桿物體")]
[Header("==========")]
public GameObject virtualJoystick;//虛擬搖桿父物體
public GameObject inside;//內(nèi)環(huán)
public GameObject outside;//外環(huán)
[Header("最大拖拽距離")]
[Header("==========")]
public float maxDragLength;//最大拖拽距離
private Vector3 virtualJoystickCenter;//虛擬軸中心
private void Update()
{
//如果限制虛擬軸位置并且虛擬軸位置超出了限制范圍則不進行任何操作
if (restrictVirtualJoystickPos && JudgeIsValidRange() == false)
{
return;
}
//更新顯示
UpdateShow();
//更新虛擬搖桿位置
if (Input.GetMouseButtonDown(0))
{
UpdateVirtualJoystickPos();
}
else if (Input.GetMouseButtonUp(0))
{
inside.transform.position = virtualJoystickCenter;
}
//更新內(nèi)環(huán)位置(限制拖拽范圍)
if (Input.GetMouseButton(0))
{
UpdateInsidePos();
}
}
/// <summary>
/// 更新顯示
/// </summary>
private void UpdateShow()
{
if (alwaysVisible)
{
inside.SetActive(true);
outside.SetActive(true);
}
else if (alwaysVisible == false)
{
if (Input.GetMouseButtonDown(0))
{
inside.SetActive(true);
outside.SetActive(true);
}
if (Input.GetMouseButtonUp(0))
{
inside.SetActive(false);
outside.SetActive(false);
}
}
}
/// <summary>
/// 更新虛擬搖桿位置
/// </summary>
private void UpdateVirtualJoystickPos()
{
//得到虛擬軸的中心位置
virtualJoystickCenter = ScreenToWorld(Input.mousePosition);
//設(shè)置虛擬軸的位置
virtualJoystick.transform.position = virtualJoystickCenter;
}
/// <summary>
/// 更新內(nèi)環(huán)位置
/// </summary>
private void UpdateInsidePos()
{
inside.transform.position = ScreenToWorld(Input.mousePosition);
if (Vector3.Distance(inside.transform.position, virtualJoystickCenter) > maxDragLength)
{
Vector3 normalizedPos = (inside.transform.position - virtualJoystickCenter).normalized;
inside.transform.position = normalizedPos * maxDragLength + virtualJoystickCenter;
}
}
/// <summary>
/// 判斷是否為有效的范圍
/// </summary>
/// <returns>是否為有效的范圍</returns>
private bool JudgeIsValidRange()
{
if (inside.activeInHierarchy)
{
return true;
}
Vector2 v = Input.mousePosition;
if (v.x > Screen.width / 2 || v.x < 0)
{
return false;
}
else if (v.y > Screen.height / 2 || v.y < 0)
{
return false;
}
return true;
}
/// <summary>
/// 屏幕坐標(biāo)轉(zhuǎn)世界坐標(biāo)
/// </summary>
/// <param name="screenPos">屏幕坐標(biāo)位置</param>
/// <param name="camera">相機</param>
/// <returns>轉(zhuǎn)換后的世界坐標(biāo)</returns>
public static 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;
}
/// <summary>
/// 得到運動的方向和長度
/// </summary>
/// <returns>方向和長度</returns>
public Vector3 GetDirAndLength()
{
return inside.transform.position - virtualJoystickCenter;
}
}
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
C# Socket編程實現(xiàn)簡單的局域網(wǎng)聊天器的示例代碼
這篇文章主要介紹了C# Socket編程實現(xiàn)簡單的局域網(wǎng)聊天器,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-03-03
WPF利用ValueConverter實現(xiàn)值轉(zhuǎn)換器
值轉(zhuǎn)換器在WPF開發(fā)中是非常常見的,值轉(zhuǎn)換器可以幫助我們很輕松地實現(xiàn),界面數(shù)據(jù)展示的問題。本文將通過WPF?ValueConverter實現(xiàn)簡單的值轉(zhuǎn)換器,希望對大家有所幫助2023-03-03
Ruby創(chuàng)建數(shù)組方法總結(jié)
在本篇文章里小編給大家分享了關(guān)于Ruby創(chuàng)建數(shù)組方法的知識點內(nèi)容,對戲有興趣的朋友們學(xué)習(xí)下。2019-01-01
C#中的multipart/form-data提交文件和參數(shù)
這篇文章主要介紹了C#中的multipart/form-data提交文件和參數(shù),具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-06-06

