Unity UGUI Selectable部分源码浅析

2024-01-30 09:44

本文主要是介绍Unity UGUI Selectable部分源码浅析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

private readonly List<CanvasGroup> m_CanvasGroupCache = new List<CanvasGroup>();protected override void OnCanvasGroupChanged(){//判断父节点中是否允许交互var groupAllowInteraction = true;Transform t = transform;while (t != null){t.GetComponents(m_CanvasGroupCache);bool shouldBreak = false;for (var i = 0; i < m_CanvasGroupCache.Count; i++){//如果父节点中的canvasgroup不允许交互,那么断开循环,设置falseif (!m_CanvasGroupCache[i].interactable){groupAllowInteraction = false;shouldBreak = true;}// 如果此节点设置了“忽略父设置”,那么也直接断开if (m_CanvasGroupCache[i].ignoreParentGroups)shouldBreak = true;}if (shouldBreak)break;t = t.parent;}if (groupAllowInteraction != m_GroupsAllowInteraction){m_GroupsAllowInteraction = groupAllowInteraction;OnSetProperty();}}

在canvasGroup属性改变时调用OnCanvasGroupChanged

public virtual bool IsInteractable(){return m_GroupsAllowInteraction && m_Interactable;}

物体的可交互属性需要一起判断m_GroupsAllowInteraction 和 m_Interactable,即自身或父物体上的canvasGroup是否允许交互,以及自身是否允许交互

protected override void OnEnable(){base.OnEnable();if (s_IsDirty)RemoveInvalidSelectables();  //如果标记为脏,就从可选择物体数组中移除m_WillRemove = false;if (s_SelectableCount == s_Selectables.Length)   //如果可选择物体数组满了,则扩容两倍{Selectable[] temp = new Selectable[s_Selectables.Length * 2];Array.Copy(s_Selectables, temp, s_Selectables.Length);s_Selectables = temp;}s_Selectables[s_SelectableCount++] = this;isPointerDown = false;DoStateTransition(currentSelectionState, true);  //设置初始状态}
protected override void OnTransformParentChanged(){base.OnTransformParentChanged();// 如果父物体有改变,需要判断自身和父物体的canvasGroup属性是否有改变OnCanvasGroupChanged();}
private void OnSetProperty(){
#if UNITY_EDITORif (!Application.isPlaying)DoStateTransition(currentSelectionState, true);else
#endifDoStateTransition(currentSelectionState, false);}

设置状态属性,如果是Editor模式,如果正在运行,那么立即改变,没有过渡,如果非Editor模式,那么有过渡

        protected override void OnDisable(){m_WillRemove = true;s_IsDirty = true;InstantClearState();  //重置各种状态base.OnDisable();}

取消激活时,将自身标记为脏

private static void RemoveInvalidSelectables(){for (int i = s_SelectableCount - 1; i >= 0; --i){// Swap last element in array with element to be removedif (s_Selectables[i] == null || s_Selectables[i].m_WillRemove)s_Selectables[i] = s_Selectables[--s_SelectableCount];}s_IsDirty = false;}

将元素从可选择物体数组列表中移除,同时标记为脏

protected SelectionState currentSelectionState{get{if (!IsInteractable())return SelectionState.Disabled;if (isPointerDown)return SelectionState.Pressed;if (hasSelection)return SelectionState.Selected;if (isPointerInside)return SelectionState.Highlighted;return SelectionState.Normal;}}

获取当前的选择状态

protected virtual void InstantClearState(){string triggerName = m_AnimationTriggers.normalTrigger;isPointerInside = false;isPointerDown = false;hasSelection = false;switch (m_Transition){case Transition.ColorTint:StartColorTween(Color.white, true);break;case Transition.SpriteSwap:DoSpriteSwap(null);break;case Transition.Animation:TriggerAnimation(triggerName);break;}}

即时清除状态,将Selectable的状态还原

protected virtual void DoStateTransition(SelectionState state, bool instant){if (!gameObject.activeInHierarchy)return;Color tintColor;Sprite transitionSprite;string triggerName;switch (state){case SelectionState.Normal:tintColor = m_Colors.normalColor;transitionSprite = null;triggerName = m_AnimationTriggers.normalTrigger;break;case SelectionState.Highlighted:tintColor = m_Colors.highlightedColor;transitionSprite = m_SpriteState.highlightedSprite;triggerName = m_AnimationTriggers.highlightedTrigger;break;case SelectionState.Pressed:tintColor = m_Colors.pressedColor;transitionSprite = m_SpriteState.pressedSprite;triggerName = m_AnimationTriggers.pressedTrigger;break;case SelectionState.Selected:tintColor = m_Colors.selectedColor;transitionSprite = m_SpriteState.selectedSprite;triggerName = m_AnimationTriggers.selectedTrigger;break;case SelectionState.Disabled:tintColor = m_Colors.disabledColor;transitionSprite = m_SpriteState.disabledSprite;triggerName = m_AnimationTriggers.disabledTrigger;break;default:tintColor = Color.black;transitionSprite = null;triggerName = string.Empty;break;}switch (m_Transition){case Transition.ColorTint:StartColorTween(tintColor * m_Colors.colorMultiplier, instant);break;case Transition.SpriteSwap:DoSpriteSwap(transitionSprite);break;case Transition.Animation:TriggerAnimation(triggerName);break;}}

状态切换方法

public Selectable FindSelectable(Vector3 dir){dir = dir.normalized;Vector3 localDir = Quaternion.Inverse(transform.rotation) * dir;Vector3 pos = transform.TransformPoint(GetPointOnRectEdge(transform as RectTransform, localDir));float maxScore = Mathf.NegativeInfinity;Selectable bestPick = null;if (s_IsDirty)RemoveInvalidSelectables();for (int i = 0; i < s_SelectableCount; ++i){Selectable sel = s_Selectables[i];if (sel == this)continue;if (!sel.IsInteractable() || sel.navigation.mode == Navigation.Mode.None)continue;#if UNITY_EDITOR除了运行时使用,FindSelectable被自定义编辑器用来在不同的可选项之间绘制箭头。对于场景视图摄像机,只考虑同一阶段的可选对象。if (Camera.current != null && !UnityEditor.SceneManagement.StageUtility.IsGameObjectRenderedByCamera(sel.gameObject, Camera.current))continue;
#endifvar selRect = sel.transform as RectTransform;Vector3 selCenter = selRect != null ? (Vector3)selRect.rect.center : Vector3.zero;Vector3 myVector = sel.transform.TransformPoint(selCenter) - pos;//值就是沿着这个方向的距离float dot = Vector3.Dot(dir, myVector);//跳过方向错误或距离为0的元素这也确保了下面的评分公式不会有除数为零的误差if (dot <= 0)continue;// This scoring function has two priorities:// - Score higher for positions that are closer.// - Score higher for positions that are located in the right direction.// This scoring function combines both of these criteria.// It can be seen as this://   Dot (dir, myVector.normalized) / myVector.magnitude// The first part equals 1 if the direction of myVector is the same as dir, and 0 if it's orthogonal.// The second part scores lower the greater the distance is by dividing by the distance.// The formula below is equivalent but more optimized.//// If a given score is chosen, the positions that evaluate to that score will form a circle// that touches pos and whose center is located along dir. A way to visualize the resulting functionality is this:// From the position pos, blow up a circular balloon so it grows in the direction of dir.// The first Selectable whose center the circular balloon touches is the one that's chosen.float score = dot / myVector.sqrMagnitude;if (score > maxScore){maxScore = score;bestPick = sel;}}return bestPick;}

此方法用来寻找临近的可选择对象

public virtual void OnMove(AxisEventData eventData){switch (eventData.moveDir){case MoveDirection.Right:Navigate(eventData, FindSelectableOnRight());break;case MoveDirection.Up:Navigate(eventData, FindSelectableOnUp());break;case MoveDirection.Left:Navigate(eventData, FindSelectableOnLeft());break;case MoveDirection.Down:Navigate(eventData, FindSelectableOnDown());break;}}

当焦点移动到另一个可选择对象时,调用此方法,确定选择的对象

protected bool IsHighlighted(){if (!IsActive() || !IsInteractable())return false;return isPointerInside && !isPointerDown && !hasSelection;}

判断是否高亮,即鼠标移上去的时候,需判断三个状态,鼠标在UI范围内,没有点下,没有选择

protected bool IsPressed(){if (!IsActive() || !IsInteractable())return false;return isPointerDown;}

是否按下

private void EvaluateAndTransitionToSelectionState(){if (!IsActive() || !IsInteractable())return;DoStateTransition(currentSelectionState, false);}

执行状态改变

public virtual void OnPointerDown(PointerEventData eventData){if (eventData.button != PointerEventData.InputButton.Left)  //如果不是鼠标左键,returnreturn;//如果可以交互,导航模式不为空,且EventSystem存在,那么设置EventSystem的当前选择物体if (IsInteractable() && navigation.mode != Navigation.Mode.None && EventSystem.current != null)EventSystem.current.SetSelectedGameObject(gameObject, eventData);isPointerDown = true;EvaluateAndTransitionToSelectionState();}

鼠标按下回调

public virtual void OnPointerUp(PointerEventData eventData){if (eventData.button != PointerEventData.InputButton.Left)return;isPointerDown = false;EvaluateAndTransitionToSelectionState();}

鼠标抬起回调

public virtual void OnPointerEnter(PointerEventData eventData){isPointerInside = true;EvaluateAndTransitionToSelectionState();}

鼠标进入回调

public virtual void OnPointerExit(PointerEventData eventData){isPointerInside = false;EvaluateAndTransitionToSelectionState();}

鼠标离开回调

public virtual void OnSelect(BaseEventData eventData){hasSelection = true;EvaluateAndTransitionToSelectionState();}

选择回调

public virtual void OnDeselect(BaseEventData eventData){hasSelection = false;EvaluateAndTransitionToSelectionState();}

取消选择回调

public virtual void Select(){if (EventSystem.current == null || EventSystem.current.alreadySelecting)return;EventSystem.current.SetSelectedGameObject(gameObject);}

直接确定EventSystem的选择物体

这篇关于Unity UGUI Selectable部分源码浅析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/659838

相关文章

浅析如何保证MySQL与Redis数据一致性

《浅析如何保证MySQL与Redis数据一致性》在互联网应用中,MySQL作为持久化存储引擎,Redis作为高性能缓存层,两者的组合能有效提升系统性能,下面我们来看看如何保证两者的数据一致性吧... 目录一、数据不一致性的根源1.1 典型不一致场景1.2 关键矛盾点二、一致性保障策略2.1 基础策略:更新数

8种快速易用的Python Matplotlib数据可视化方法汇总(附源码)

《8种快速易用的PythonMatplotlib数据可视化方法汇总(附源码)》你是否曾经面对一堆复杂的数据,却不知道如何让它们变得直观易懂?别慌,Python的Matplotlib库是你数据可视化的... 目录引言1. 折线图(Line Plot)——趋势分析2. 柱状图(Bar Chart)——对比分析3

浅析Java如何保护敏感数据

《浅析Java如何保护敏感数据》在当今数字化时代,数据安全成为了软件开发中至关重要的课题,本文将深入探讨Java安全领域,聚焦于敏感数据保护的策略与实践,感兴趣的小伙伴可以了解下... 目录一、Java 安全的重要性二、敏感数据加密技术(一)对称加密(二)非对称加密三、敏感数据的访问控制(一)基于角色的访问

Android实现一键录屏功能(附源码)

《Android实现一键录屏功能(附源码)》在Android5.0及以上版本,系统提供了MediaProjectionAPI,允许应用在用户授权下录制屏幕内容并输出到视频文件,所以本文将基于此实现一个... 目录一、项目介绍二、相关技术与原理三、系统权限与用户授权四、项目架构与流程五、环境配置与依赖六、完整

浅析如何使用xstream实现javaBean与xml互转

《浅析如何使用xstream实现javaBean与xml互转》XStream是一个用于将Java对象与XML之间进行转换的库,它非常简单易用,下面将详细介绍如何使用XStream实现JavaBean与... 目录1. 引入依赖2. 定义 JavaBean3. JavaBean 转 XML4. XML 转 J

Android实现定时任务的几种方式汇总(附源码)

《Android实现定时任务的几种方式汇总(附源码)》在Android应用中,定时任务(ScheduledTask)的需求几乎无处不在:从定时刷新数据、定时备份、定时推送通知,到夜间静默下载、循环执行... 目录一、项目介绍1. 背景与意义二、相关基础知识与系统约束三、方案一:Handler.postDel

Java 正则表达式URL 匹配与源码全解析

《Java正则表达式URL匹配与源码全解析》在Web应用开发中,我们经常需要对URL进行格式验证,今天我们结合Java的Pattern和Matcher类,深入理解正则表达式在实际应用中... 目录1.正则表达式分解:2. 添加域名匹配 (2)3. 添加路径和查询参数匹配 (3) 4. 最终优化版本5.设计思

一文详解如何在Python中从字符串中提取部分内容

《一文详解如何在Python中从字符串中提取部分内容》:本文主要介绍如何在Python中从字符串中提取部分内容的相关资料,包括使用正则表达式、Pyparsing库、AST(抽象语法树)、字符串操作... 目录前言解决方案方法一:使用正则表达式方法二:使用 Pyparsing方法三:使用 AST方法四:使用字

浅析Java中如何优雅地处理null值

《浅析Java中如何优雅地处理null值》这篇文章主要为大家详细介绍了如何结合Lambda表达式和Optional,让Java更优雅地处理null值,感兴趣的小伙伴可以跟随小编一起学习一下... 目录场景 1:不为 null 则执行场景 2:不为 null 则返回,为 null 则返回特定值或抛出异常场景

Java调用C++动态库超详细步骤讲解(附源码)

《Java调用C++动态库超详细步骤讲解(附源码)》C语言因其高效和接近硬件的特性,时常会被用在性能要求较高或者需要直接操作硬件的场合,:本文主要介绍Java调用C++动态库的相关资料,文中通过代... 目录一、直接调用C++库第一步:动态库生成(vs2017+qt5.12.10)第二步:Java调用C++