添加链接
link之家
链接快照平台
  • 输入网页链接,自动生成快照
  • 标签化管理网页链接

Unity ScrollView实现自动吸附效果

作者:Hello Bug.

这篇文章主要为大家详细介绍了Unity ScrollView实现自动吸附效果,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

本文实例为大家分享了Unity ScrollView实现自动吸附效果的具体代码,供大家参考,具体内容如下

一、效果演示

二、实现思路

通过使用UGUI的拖拽接口,在拖拽结束时比较当前滑动框的NormalizedPositon与每一页的NormalizedPositon值,找到距离当前拖拽结束位置最近的页并缓慢滑动过去

三、使用说明

——此功能脚本是对ScrollView的扩展,所以必须添加UGUI提供的基础Scroll View
——Content上必须添加GridLayoutGroup组件并添加所有列表中的项(不是动态添加),只是为了方便满足布局需求(我在代码中对startCorner、startAxis、childAlignment和constraintCount进行了限制,不需要对其设置)
——不能添加Content Size Fitter组件
——测试出适合的视为滑动一页的距离和视为滑动多页的距离数值并填入即可

四、完整代码

将AutoAdsorbScrollView脚本挂载到ScrollView上

using UnityEngine; using UnityEngine.UI; using UnityEngine.EventSystems; /// <summary> /// 自动吸附的滑动列表 /// </summary> public class AutoAdsorbScrollView : MonoBehaviour, IBeginDragHandler, IEndDragHandler private ScrollRect scrollRect;//滑动框组件 private RectTransform content;//滑动框的Content private GridLayoutGroup layout;//布局组件 private int totalPage; //总页数 private int curPage; //当前页的下标 private float[] eachPageNUPos; //每页的NormalizedPositon的值 private float targetNUPos; //目标页的NormalizedPositon的值 private Vector2 beginMousePos; //鼠标开始按下的位置 private Vector2 endMousePos; //鼠标结束按下的位置 private bool isDrag; //是否在拖拽 [Header("是否可以滑动多页")] public bool sliderMultPage; [Header("视为滑动一页的距离")] [Space(25)] public float sliderOnePageDis; [Header("视为滑动多页的距离")] public float sliderMultPageDis; [Header("缓动到目标页的持续时间")] public float duration; #region Init private void Awake() scrollRect = GetComponent<ScrollRect>(); content = scrollRect.content; layout = content.GetComponent<GridLayoutGroup>(); Init();//初始化 /// <summary> /// 初始化 /// </summary> private void Init() totalPage = content.childCount; SetContentSize();//设置Content大小 CalcEachPageNUPos();//计算每一页的NormalizedPositon值 SetLayout();//设置布局 /// <summary> /// 设置Content大小 /// </summary> private void SetContentSize() content.sizeDelta = new Vector2 layout.padding.right + layout.padding.left + (totalPage - 1) * (layout.cellSize.x + layout.spacing.x) - layout.spacing.x, content.sizeDelta.y /// <summary> /// 计算每一页的NormalizedPositon值 /// </summary> private void CalcEachPageNUPos() float tempNUPos = 0; eachPageNUPos = new float[totalPage]; for (int i = 0; i < totalPage; i++) eachPageNUPos[i] = tempNUPos; tempNUPos += 1f / (totalPage - 1); /// <summary> /// 设置布局 /// </summary> private void SetLayout() scrollRect.horizontal = true; scrollRect.vertical = false; layout.padding.right = layout.padding.left; layout.startCorner = GridLayoutGroup.Corner.UpperLeft; layout.childAlignment = TextAnchor.MiddleCenter; layout.constraintCount = 1; #endregion #region Main /// <summary> /// 拖拽开始 /// </summary> public void OnBeginDrag(PointerEventData eventData) isDrag = true; beginMousePos = Input.mousePosition; /// <summary> /// 拖拽结束 /// </summary> /// <param name="eventData"></param> public void OnEndDrag(PointerEventData eventData) isDrag = false; coe = 0; endMousePos = Input.mousePosition; Vector2 offset = endMousePos - beginMousePos; Debug.Log("滑动距离为:" + offset); if (sliderMultPage) //单页滑动 if (Mathf.Abs(offset.x) >= sliderOnePageDis && Mathf.Abs(offset.x) < sliderMultPageDis) float tempHorizontalNUPos = scrollRect.horizontalNormalizedPosition; FindNearlyPage(tempHorizontalNUPos); //多页滑动 else if (Mathf.Abs(offset.x) >= sliderMultPageDis) if (offset.x > 0) curPage = 0; else if (offset.x < 0) curPage = totalPage - 1; //单页滑动 if (Mathf.Abs(offset.x) >= sliderOnePageDis) float tempHorizontalNUPos = scrollRect.horizontalNormalizedPosition; FindNearlyPage(tempHorizontalNUPos); targetNUPos = eachPageNUPos[curPage]; private float coe;//比例系数 private void Update() if (isDrag) return; coe += Time.deltaTime / duration; scrollRect.horizontalNormalizedPosition = Mathf.Lerp(scrollRect.horizontalNormalizedPosition, targetNUPos, coe); #endregion #region Tool /// <summary> /// 寻找距离当前NormalizedPositon最近的页 /// </summary> private void FindNearlyPage(float tempHorizontalNUPos) float minOffset = Mathf.Abs(eachPageNUPos[0] - tempHorizontalNUPos); for (int i = 0; i < totalPage; i++) float tempHorizontalOffset = Mathf.Abs(eachPageNUPos[i] - tempHorizontalNUPos); if (tempHorizontalOffset <= minOffset) minOffset = tempHorizontalOffset; curPage = i; #endregion

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

您可能感兴趣的文章:
  • C#使用Thrift作为RPC框架入门详细教程
    C#使用Thrift作为RPC框架入门详细教程
    2021-11-11
  • c#抽签系统的实现示例
    c#抽签系统的实现示例
    2021-10-10
  • Unity EasyTouch摇杆插件使用示例详解
    Unity EasyTouch摇杆插件使用示例详解
    2021-10-10
  • Unity 制作一个分数统计系统
    Unity 制作一个分数统计系统
    2021-10-10
  • C# StackExchange.Redis 用法汇总
    C# StackExchange.Redis 用法汇总
    2021-10-10
  • 基于C#实现端口扫描器(单线程和多线程)
    基于C#实现端口扫描器(单线程和多线程)
    2021-10-10
  • C# Quartzs定时器的使用教程
    C# Quartzs定时器的使用教程
    2021-10-10
  • 美国设下计谋,用娘炮文化重塑日本,已影响至中国
    美国设下计谋,用娘炮文化重塑日本,已影响至中国
    2021-11-19
  • 时空伴随者是什么意思?时空伴随者介绍
    时空伴随者是什么意思?时空伴随者介绍
    2021-11-09
  • 工信部称网盘企业免费用户最低速率应满足基本下载需求,天翼云盘回应:坚决支持,始终
    工信部称网盘企业免费用户最低速率应满足基本下载需求,天翼云盘回应:坚决支持,始终
    2021-11-05
  • 2022年放假安排出炉:五一连休5天 2022年所有节日一览表
    2022年放假安排出炉:五一连休5天 2022年所有节日一览表
    2021-10-26
  • 电脑版 - 返回首页

    2006-2023 脚本之家 JB51.Net , All Rights Reserved.
    苏ICP备14036222号