如今手机游戏玩法多种多样,尤其使用虚拟摇杆进行格斗类游戏开发或者是MMORPG成为了主流的开发方式,可能不少人都会为了制作一个完善的虚拟摇杆感到烦恼,一次又不少人选择使用插件来制作虚拟摇杆。
Momo大神在不久前才写了一篇用UGUI制作虚拟摇杆的,有兴趣的同学可以到这里看看原文,原文地址:http://www.xuanyusong.com/archives/3924
今天刚好公司向里面需要用到这个功能,因此我也研究了一下Momo这篇文章,发现里面有一个算是bug的问题因此我就在Momo大神的代码上进行修改,修复了这个问题而且也添加一下功能方便大家去拓展和使用。
先来看看我和我的小伙伴发现的bug吧。
下图两张图片是Momo大神的写的UGUI虚拟摇杆的小Demo,我们来看看其效果吧:
图片一:
图片二:
比较一下两张图片你会发现有一个致命问题小球的偏移量出问题了,这种情况只出现在虚拟摇杆距离屏幕边距较近的时候会出现这个问题,远离边距的时候就不会出现这个问题了,来看看远离边距时候的效果吧。如下图所示:
这样的限制对于摇杆功能来说并不会有太大的影响,但是UI的设计上还是会出现一些问题和麻烦的。因此我今天就是为了修复这个问题而对UGUI虚拟摇杆进行了研究,花了一点时间进行了bug的修复,然后拓写了它的功能,方便我们来获取摇杆的偏移量来控制人物移动。
先来看看我实现的效果吧,废话不多说上图!
上面的图片可以看出来,虚拟摇杆已经贴着屏幕边框了,但是偏移量并没有减少。
我也不说多少神马了因为实在是太简单了,我就直接上代码吧
using UnityEngine;
using System.Collections;
using UnityEngine.EventSystems;
using UnityEngine.UI;
public class ScrollCircle :ScrollRect
{
public float recoveryTime = 0.1f;
protected float mRadius;
protected bool isOnEndDrag = false;
protected Vector3 offsetVector3 = Vector3.zero;
void Start()
{
inertia = false;
movementType = MovementType.Unrestricted;
//计算摇杆块的半径
mRadius = (transform as RectTransform).sizeDelta.x * 0.5f;
}
public override void OnScroll(PointerEventData data)
{
}
public override void OnDrag (PointerEventData eventData)
{
base.OnDrag (eventData);
isOnEndDrag = false;
var contentPostion = this.content.anchoredPosition;
if (contentPostion.magnitude > mRadius){
contentPostion = contentPostion.normalized * mRadius ;
SetContentAnchoredPosition(contentPostion);
}
}
public override void OnEndDrag(PointerEventData eventData)
{
base.OnEndDrag(eventData);
if (!isOnEndDrag)
isOnEndDrag = true;
}
void Update()
{
UpdateContent();
}
/// <summary>
/// 摇杆小球复位
/// </summary>
public void UpdateContent()
{
if (isOnEndDrag)
{
if (content.localPosition == Vector3.zero)
isOnEndDrag = false;
float x = Mathf.Lerp(content.localPosition.x, 0.0f, recoveryTime);
float y = Mathf.Lerp(content.localPosition.y, 0.0f, recoveryTime);
content.localPosition = new Vector3(x, y, content.localPosition.z);
}
CalculateOffset();
}
/// <summary>
/// 计算偏移量
/// </summary>
private void CalculateOffset()
{
offsetVector3 = content.localPosition / mRadius;
}
/// <summary>
/// 获取偏移量大小
/// 偏移量范围是[-1,1]
/// </summary>
/// <returns></returns>
public Vector3 GetOffsetVector3()
{
return offsetVector3;
}
}
这里要注意一点,我在start函数里面修改了movementType,如果没有修改movementType参数的同学是没办法实现这个功能的~切记切记!!
最后上一张图片,红框的位置是要特别注意的地方!
今天写了两篇文章好累啊,我会尽量把自己学到研究到的东西都写出来,可能会很乱很杂大家别介意哈....哈哈哈哈~~
-----Begonia