zl程序教程

您现在的位置是:首页 >  其他

当前栏目

Unity 中 UGUI 事件顺序

事件 Unity 顺序 Ugui
2023-09-11 14:20:50 时间

//15.输入处理---------------------------------------------------------------------------------------------

继承关系:

BaseInputModule<---PointerInputModule<----TouchInputModule

BaseInputModule<---PointerInputModule<----StandaloneInputModule

函数调用关系:

EventSystem.Update()---子类.Process()---子类.ProcessMouseEvent()或 子类.ProcessMouseEvents()---子类.GetMousePointerEventData()

 

EventSystem.Process() 多态.

 

输入管理: EventSystem物体上的Standalone Input Module,TouchInputModule组件

 

//16. 射线检测-------------------------------------------------------------------------------------------

继承关系:

BaseRaycaster<---PhysicsRaycaster<---Physics2DRaycaster

BaseRaycaster<---GraphicRaycaster

 

函数调用关系:

EventSystem.RaycastAll()---子类.Raycast()---EventSystem.RaycastComparer()

 

EventSystem.Raycast() 多态. 进行射线检测.

 

射线管理: Canvas画布上的Graphic Raycaster组件

[RequireComponent(typeof(Canvas))]

public class GraphicRaycaster : BaseRaycaster

{...}

这个脚本必须挂在Canvas上.UI元素必须具有继承自Graphic的组件.也就是可视的UI元素.

GraphicRaycaster组件的成员属性:

Ignore Reversed Graphic  忽略反方向图形. 即.图形面对前方时.才参与射线检测.

Blocking Objects  屏蔽对象

渲染模式不为ScreenSpaceOverlay时起作用.屏蔽指定类型的对象.使他们不参与射线检测.

None 不屏蔽任何物理对象

TwoD 屏蔽2d物理对象.(具有2d碰撞体的对象)

ThreeD 屏蔽3d物理对象.(具有3d碰撞体的对象)

All 屏蔽所有物理对象

Blocking Mask 屏蔽层(layer)

    渲染模式不为ScreenSpaceOverlay,且locking objects不为None时起作用.

    使屏蔽对象(Blocking objects)的物体,不参与射线检测.

 

注意:添加的碰撞器大小要和物体一样大.或者比物体本身大.创建的碰撞体默认是长宽都是1.狠小狠难点中.

 

 

 

射线穿透的顺序:

 

EventSystem.cs中

射线处理,是把所有的装到链表里.然后排序. 

public void RaycastAll(PointerEventData eventData, List<RaycastResult> raycastResults)

{

    raycastResults.Clear();

    var modules = RaycasterManager.GetRaycasters();

    for (int i = 0; i < modules.Count; ++i)
    
    {

        var module = modules[i];

        if (module == null || !module.IsActive())

            continue;


        module.Raycast(eventData, raycastResults);

    }


    raycastResults.Sort(s_RaycastComparer);

}

 

排序规则函数  RaycastComparer

private static readonly Comparison<RaycastResult> s_RaycastComparer = RaycastComparer;


private static int RaycastComparer(RaycastResult lhs, RaycastResult rhs)

{

    if (lhs.module != rhs.module)

    {...}

    if (lhs.sortingLayer != rhs.sortingLayer)

        return rhs.sortingLayer.CompareTo(lhs.sortingLayer);

    
    if (lhs.sortingOrder != rhs.sortingOrder)

        return rhs.sortingOrder.CompareTo(lhs.sortingOrder);


    if (lhs.depth != rhs.depth)

        return rhs.depth.CompareTo(lhs.depth);

    
    if (lhs.distance != rhs.distance)

        return lhs.distance.CompareTo(rhs.distance);


    return lhs.index.CompareTo(rhs.index);

}

射线穿透排序

EventSystem.RaycastComparer

对射线穿透的所有未屏蔽的物体进行排序

  1. 如果Canvas指定了不同摄像机,则按摄像机深度排序.

  2. 按sortOrder顺序排序

  3. 按renderOrder顺序排序.

  4. 按depth排序.凡是继承自Graphic的类都有depth属性.在PhysicsRaycaster,Physic2DRaycaster两种射线模式下,该值为0.

  5. 按distance顺序排序.射线发出点到物体的具体.

  6. 按index顺序排序. 不管是哪种射线模式,或者是混合射线模式,都会将射线碰撞到的UI物体放入一个链表或者数组,这里的index就是他的下标索引值.

 

 

组件Canvas Group. 可以管理Canvas.也可以控制Canvas内UI元素.

属性:

alpha  透明度

interactable 是否可交互

Blocks Raycasts 控制是否穿透

Ignore parent group是否忽略父物体的Canvas Group组件效果.

 

//17. 事件分发-------------------------------------------------------------------------------------------

函数调用关系:

BaseInputModule.FindFirstRaycast (...)取出射线碰撞到的第一个物体,进行事件处理.

ExecuteEvents.ExecuteHierarchy (...)从筛选出的第一个物体自身开始,一级一级往上获取父物体.直到有组件(实现了该事件接口的组件)接受事件为止.

protected static RaycastResult FindFirstRaycast(List<RaycastResult> candidates)

{

    for (var i = 0; i < candidates.Count; ++i)

{

if (candidates[i].gameObject == null)

    continue;


    return candidates[i];

}

return new RaycastResult();

}


public static GameObject ExecuteHierarchy<T>(GameObject root, BaseEventData eventData, EventFunction<T> callbackFunction) where T : IEventSystemHandler

{

    GetEventChain(root, s_InternalTransformList);


for (var i = 0; i < s_InternalTransformList.Count; i++)

{

    var transform = s_InternalTransformList[i];

    if (Execute(transform.gameObject, eventData, callbackFunction))

    return transform.gameObject;

}

    return null;

}

 

测试:

//脚本  rayTest.cs


 

using UnityEngine;

using System.Collections;

using UnityEngine.EventSystems;


public class rayTest : MonoBehaviour ,IPointerClickHandler

{

    public void OnPointerClick (PointerEventData eventData)

    {

        Debug.Log (name);

        Debug.Log (eventData.pointerCurrentRaycast.gameObject);

    }

}

 

射线第一个扫到的是Text1物体但是他上面没有实现点击事件接口.

所以从Text1开始一级一级找父物体看看父物体祖宗物体上有没有实现这个事件

找到Button上面rayTest脚本中有实现 IPointerClickHandler点击事件接口.

所以就调用这个事件的响应函数OnPointerClick (...)