Unity3d中Tab控件的实现

1.按如下结构建立一个Tabs

2.Tabs上添加Toggle Group组件

3.Tab上添加Toggle、ToggleHighlightStaysActiveAfterLosingFocus

4.Page上添加Canvas Group

5.在Menu上添加TabMenu,系统会自动识别,无需手动赋值(识别错误的话见下文)。这样就完成了Tab控件的制作

6.如果Page下有Toggle,TabMenu会识别错误,需要手动赋值。

解决方法:注释TabMenu.cs以下代码,再手动赋值

7.设定初始选中项,将对应的Tab下的Toggle的isOn勾上,其余Tab的取消勾选就可以了。

附:

TabMenu.cs代码

cs 复制代码
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.Events;
using UnityEngine.UI;

namespace ChristinaCreatesGames.UI
{
    public class TabMenu : MonoBehaviour
    {
        [Header("Current Index")]
        [SerializeField] private int pageIndex = 0;
        
        [Header("Components")]
        [SerializeField] private ToggleGroup toggleGroup;
        [SerializeField] private List<Toggle> tabs = new List<Toggle>();
        [SerializeField] private List<CanvasGroup> pages = new List<CanvasGroup>();
        
        [Header("Event to call")]
        public UnityEvent<int> OnPageIndexChanged;

        private void Initialize()
        {
            toggleGroup = GetComponentInChildren<ToggleGroup>();
            
            tabs.Clear();
            pages.Clear();
            
            tabs.AddRange(GetComponentsInChildren<Toggle>());
            pages.AddRange(GetComponentsInChildren<CanvasGroup>());
        }

        private void Reset()
        {
            Initialize();
        }

        private void OnValidate()
        {
            Initialize();
            OpenPage(pageIndex);
            tabs[pageIndex].SetIsOnWithoutNotify(true);
        }
        
        private void Awake()
        {
            foreach (var toggle in tabs)
            {
                toggle.onValueChanged.AddListener(CheckForTab);
                toggle.group = toggleGroup;
            }
        }

        private void OnDestroy()
        {
            foreach (var toggle in tabs)
            {
                toggle.onValueChanged.RemoveListener(CheckForTab);
            }
        }
        
        private void CheckForTab(bool value)
        {
            for (int i = 0; i < tabs.Count; i++)
            {
                if (!tabs[i].isOn) continue;
                pageIndex = i;
            }
            
            OpenPage(pageIndex);
        }
        
        private void OpenPage(int index)
        {
            EnsureIndexIsInRange(index);
            
            for (int i = 0; i < pages.Count; i++)
            {
                bool isActivePage = (i == pageIndex);
                
                pages[i].alpha = isActivePage ? 1.0f : 0.0f;
                pages[i].interactable = isActivePage;
                pages[i].blocksRaycasts = isActivePage;
            }
            
            if (Application.isPlaying)
                OnPageIndexChanged?.Invoke(pageIndex);
        }

        private void EnsureIndexIsInRange(int index)
        {
            if (tabs.Count == 0 || pages.Count == 0)
            {
                Debug.Log("Forgot to setup Tabs or Pages");
                return;
            }
            
            pageIndex = Mathf.Clamp(index, 0, pages.Count - 1);
        }
        
        public void JumpToPage(int page)
        {
            EnsureIndexIsInRange(page);
            
            tabs[pageIndex].isOn = true;
        }
    }
}

ToggleHighlightStaysActiveAfterLosingFocus.cs代码

cs 复制代码
using UnityEngine;
using UnityEngine.UI;

namespace ChristinaCreatesGames.UI
{
    public class ToggleHighlightStaysActiveAfterLosingFocus : MonoBehaviour
    {
        [SerializeField] private Toggle toggle;
        [SerializeField] private Image  imageToKeepFocusActive;
        
        private void Reset()
        {
            toggle = GetComponent<Toggle>();
        }

        private void Awake()
        {
            toggle.onValueChanged.AddListener(OnToggleValueChanged);
            OnToggleValueChanged(toggle.isOn);
        }

        private void OnDestroy()
        {
            toggle.onValueChanged.RemoveListener(OnToggleValueChanged);
        }

        private void OnToggleValueChanged(bool isOn)
        {
            if (imageToKeepFocusActive == null) return;

            imageToKeepFocusActive.color = toggle.isOn ? toggle.colors.highlightedColor : Color.clear;
        }
    }
}

参考视频

相关推荐
雨季66612 小时前
Flutter 三端应用实战:OpenHarmony 简易“动态主题切换卡片”交互模式
flutter·ui·交互·dart
淡海水14 小时前
【节点】[Houndstooth节点]原理解析与实际应用
unity·游戏引擎·shadergraph·图形·houndstooth
雨季66614 小时前
Flutter 三端应用实战:OpenHarmony 简易“动态色盘生成器”交互模式深度解析
开发语言·前端·flutter·ui·交互
雨季66614 小时前
Flutter 三端应用实战:OpenHarmony 简易“可展开任务详情卡片”交互模式深度解析
开发语言·前端·javascript·flutter·ui·交互
GIS小小研究僧15 小时前
如何使用Photoshop扣透明底电子签名
ui·photoshop
不会代码的小测试16 小时前
UI自动化-下拉选择框多级联动情况进行选择
前端·javascript·python·ui·自动化
不会代码的小测试16 小时前
UI自动化-下拉元素被隐藏无法直接进行select选择情况解决
前端·javascript·ui·自动化
雨季66616 小时前
Flutter 三端应用实战:OpenHarmony 简易“圆形进度启动屏”交互模式深度解析
开发语言·前端·javascript·flutter·ui·交互
熊猫钓鱼>_>17 小时前
【开源鸿蒙跨平台开发先锋训练营】Day 14: React Native开发鸿蒙应用第二阶段复盘——从功能实现到工程化实践的跨越
react native·ui·开源·harmonyos·arkts·鸿蒙·flatlist
微祎_1 天前
Flutter for OpenHarmony:构建一个 Flutter 情绪日记应用,深入解析状态管理、Chip 选择器与心理健康类 UI 设计
javascript·flutter·ui