当前位置: 首页 > article >正文

【Unity3D编辑器扩展】Unity3D中实现快速切换场景功能(提高效率)

推荐阅读

  • CSDN主页
  • GitHub开源地址
  • Unity3D插件分享
  • QQ群:398291828
  • 小红书
  • 小破站

大家好,我是佛系工程师☆恬静的小魔龙☆,不定时更新Unity开发技巧,觉得有用记得一键三连哦。

一、前言

当场景多了之后,并且场景放到不同的文件夹后,切换场景是一个痛苦的过程(懂的都懂)。

今天就分享一个在Unity3D编辑器中快速切换场景的小技巧,基于Unity3D编辑器扩展进行实现。

接下来,就开始了。

二、正文

2-1、实现过程

首先,新建一个Editor文件夹,只有在这个文件夹内代码才有用,而且也不会被打包进去。

在这里插入图片描述
在Editor文件夹内新建脚本,名字随意,我名字命名为:EditorSwithScene

编辑代码:

using System.Reflection;
using UnityEditor;
using UnityEngine;
using System;
using System.Collections.Generic;
using UnityEditor.SceneManagement;
using UnityEngine.SceneManagement;
#if UNITY_2019_1_OR_NEWER
using UnityEngine.UIElements;
#else
using UnityEngine.Experimental.UIElements;
#endif

/// <summary>
/// 扩展Unity的按钮栏
/// </summary>
[InitializeOnLoad]
public static class EditorSwithScene
{
    public static Type m_toolbarType = typeof(Editor).Assembly.GetType("UnityEditor.Toolbar");
    public static ScriptableObject m_currentToolbar;

    private static GUIContent switchSceneBtContent;
    private static List<string> sceneAssetList;

    static EditorSwithScene()
    {
        EditorApplication.delayCall += OnUpdate;
    }

    [InitializeOnLoadMethod]
    static void Init()
    {
        sceneAssetList = new List<string>();
        var curOpenSceneName = EditorSceneManager.GetActiveScene().name;
        switchSceneBtContent = EditorGUIUtility.TrTextContentWithIcon(string.IsNullOrEmpty(curOpenSceneName) ? "Switch Scene" : curOpenSceneName, "切换场景", "UnityLogo");

        EditorSceneManager.sceneOpened += OnSceneOpened;
    }

    private static void OnSceneOpened(Scene scene, OpenSceneMode mode)
    {
        switchSceneBtContent.text = scene.name;
    }

    static int m_toolCount;
    public static void OnUpdate()
    {
        if (m_currentToolbar == null)
        {
            // 查找系统自带的Toolbar
            var toolbars = Resources.FindObjectsOfTypeAll(m_toolbarType);
            m_currentToolbar = toolbars.Length > 0 ? (ScriptableObject)toolbars[0] : null;

            if (m_currentToolbar != null)
            {
                // 固定写法
                var root = m_currentToolbar.GetType().GetField("m_Root", BindingFlags.NonPublic | BindingFlags.Instance);
                var rawRoot = root.GetValue(m_currentToolbar);
                var mRoot = rawRoot as VisualElement;
                RegisterCallback("ToolbarZoneLeftAlign", GUILeft);
                RegisterCallback("ToolbarZoneRightAlign", GUIRight);

                void RegisterCallback(string root, Action cb)
                {
                    var toolbarZone = mRoot.Q(root);

                    var parent = new VisualElement()
                    {
                        style = 
                        {
                            flexGrow = 1,
                            flexDirection = FlexDirection.Row,
                        }
                    };
                    var container = new IMGUIContainer();
                    container.onGUIHandler += () =>
                    {
                        cb?.Invoke();
                    };
                    parent.Add(container);
                    toolbarZone.Add(parent);
                }
            }
        }
    }

    /// <summary>
    /// 绘制左侧的元素
    /// </summary>
    private static void GUILeft()
    {
        GUILayout.BeginHorizontal();
        if (GUILayout.Button("可扩展"))
        { }
        GUILayout.EndHorizontal();
    }

    /// <summary>
    /// 绘制右侧的元素
    /// </summary>
    private static void GUIRight()
    {
        GUILayout.BeginHorizontal();
        if (EditorGUILayout.DropdownButton(switchSceneBtContent, FocusType.Passive, EditorStyles.toolbarPopup, GUILayout.MaxWidth(150)))
        {
            DrawSwithSceneDropdownMenus();
        }
        GUILayout.EndHorizontal();
    }

    #region 绘制切换场景的Toolbar
    /// <summary>
    /// 绘制SwitchScene的下拉菜单
    /// </summary>
    static void DrawSwithSceneDropdownMenus()
    {
        GenericMenu popMenu = new GenericMenu();
        popMenu.allowDuplicateNames = true;
        var sceneGuids = AssetDatabase.FindAssets("t:Scene", new string[] { "Assets" });
        sceneAssetList.Clear();
        for (int i = 0; i < sceneGuids.Length; i++)
        {
            var scenePath = AssetDatabase.GUIDToAssetPath(sceneGuids[i]);
            sceneAssetList.Add(scenePath);
            string fileDir = System.IO.Path.GetDirectoryName(scenePath);
            bool isInRootDir = GetRegularPath("Assets").TrimEnd('/') == GetRegularPath(fileDir).TrimEnd('/');
            var sceneName = System.IO.Path.GetFileNameWithoutExtension(scenePath);
            string displayName = sceneName;
            if (!isInRootDir)
            {
                var sceneDir = System.IO.Path.GetRelativePath("Assets", fileDir);
                displayName = $"{sceneDir}/{sceneName}";
            }

            popMenu.AddItem(new GUIContent(displayName), false, menuIdx => { SwitchScene((int)menuIdx); }, i);
        }
        popMenu.ShowAsContext();
    }

    static string GetRegularPath(string path)
    {
        return path?.Replace('\\', '/');
    }

    /// <summary>
    /// 执行切换场景的事件
    /// </summary>
    /// <param name="menuIdx"></param>
    private static void SwitchScene(int menuIdx)
    {
        if (menuIdx >= 0 && menuIdx < sceneAssetList.Count)
        {
            var scenePath = sceneAssetList[menuIdx];
            var curScene = EditorSceneManager.GetActiveScene();
            if (curScene != null && curScene.isDirty)
            {
                int opIndex = EditorUtility.DisplayDialogComplex("警告", $"当前场景{curScene.name}未保存,是否保存?", "保存", "取消", "不保存");
                switch (opIndex)
                {
                    case 0:
                        if (!EditorSceneManager.SaveOpenScenes())
                        {
                            return;
                        }
                        break;
                    case 1:
                        return;
                }
            }
            EditorSceneManager.OpenScene(scenePath, OpenSceneMode.Single);
        }
    }
    #endregion
}

编译完成后:
在这里插入图片描述

2-2、注意点

(1)Unity3D编辑器的标题栏扩展,需要先反射获取到Toolbar:
在这里插入图片描述
(2)GUI渲染及事件执行:
在这里插入图片描述
(3)如果要执行只去切换某个文件夹内的场景,可以修改红框中的Assets比如改成Assets/Scenes
在这里插入图片描述
(4)如果不想显示根目录,可以修改这一块代码,将isInRootDir设为true
在这里插入图片描述

三、后记

如果觉得本篇文章有用别忘了点个关注,关注不迷路,持续分享更多Unity干货文章。


你的点赞就是对博主的支持,有问题记得留言:

博主主页有联系方式。

博主还有跟多宝藏文章等待你的发掘哦:

专栏方向简介
Unity3D开发小游戏小游戏开发教程分享一些使用Unity3D引擎开发的小游戏,分享一些制作小游戏的教程。
Unity3D从入门到进阶入门从自学Unity中获取灵感,总结从零开始学习Unity的路线,有C#和Unity的知识。
Unity3D之UGUIUGUIUnity的UI系统UGUI全解析,从UGUI的基础控件开始讲起,然后将UGUI的原理,UGUI的使用全面教学。
Unity3D之读取数据文件读取使用Unity3D读取txt文档、json文档、xml文档、csv文档、Excel文档。
Unity3D之数据集合数据集合数组集合:数组、List、字典、堆栈、链表等数据集合知识分享。
Unity3D之VR/AR(虚拟仿真)开发虚拟仿真总结博主工作常见的虚拟仿真需求进行案例讲解。
Unity3D之插件插件主要分享在Unity开发中用到的一些插件使用方法,插件介绍等
Unity3D之日常开发日常记录主要是博主日常开发中用到的,用到的方法技巧,开发思路,代码分享等
Unity3D之日常BUG日常记录记录在使用Unity3D编辑器开发项目过程中,遇到的BUG和坑,让后来人可以有些参考。

http://www.kler.cn/a/409346.html

相关文章:

  • DataWhale—PumpkinBook(TASK05决策树)
  • C语言指针作业
  • 美创科技入选2024数字政府解决方案提供商TOP100!
  • repmgr安装及常用运维指令
  • C++初阶(十五)--STL--list 的深度解析与全面应用
  • 硬中断关闭后的堆栈抓取方法
  • 现代大数据架构设计与实践:从数据存储到处理的全面解读
  • HarmonyOS(57) UI性能优化
  • Golang语言系列-Channel
  • 《数据结构》学习系列——图(中)
  • 基于 BP 神经网络整定的 PID 控制
  • 根据气候变化自动制定鲜花存储策略(BabyAGI)
  • MCSA --- make coding simple again
  • C#里怎么样实现多播委托?
  • AIGC-------AIGC与创意写作:威胁还是机遇?
  • [webgis 0基础到找工作]------JavaScript进阶--作用域,解构,函数 day14
  • [webgis 0基础到找工作]------JavaScript--Bom day12
  • 地平线 bev_cft_efficientnetb3 参考算法-v1.2.1
  • 如何进行高级红队测试:OpenAI的实践与方法
  • HTTPSOK ---助力阿里云免费 SSL 证书自动续期
  • 废品买卖回收管理系统|Java|SSM|Vue| 前后端分离
  • Jmeter中的定时器
  • 基于STM32F103的FreeRTOS系列(十四)·软件定时器
  • 【Excel】拆分多个sheet,为单一表格
  • 微调Helsinki-NLP-en-zh模型
  • Python爬虫:如何从1688阿里巴巴获取公司信息