UnityEditor编辑器扩展代码实现Project搜索的实现功能和切换Component等

news2025/1/10 10:49:45

反射实现切换Gameobjecect-Comp

之前介绍过Kinematic Character Controller这个插件

这个插件很容易和另外一个插件混淆,两个作者头像比较相像,而且这个插件的作者不太喜欢露脸(他现在做Dot-CharacterControl去了),几乎网上找到的都是另一个CharacterController插件

但其实这个控件例子不少的,都是走,跑和跳等例子。

而且例子内代码结构是:多个例子场景,则多个命名空间 ,每个命名空间内一个MyController(同名)

整个Kinematic Character Controller,几乎有十几个MyController

咋一看,怎么不使用继承,不太面向对象的感觉,太不专业;但实际上也和面向对象的使用无差

我们的需求来了:一个GameObject 包含一个逻辑 Mono,是否可以直接右键替换?

??切换的同时重点是如何把Mono的SerilizeField(一些关联go)也一并替换,这就是用到反射了

否则,就得一个个场景切换代码,比较麻烦

所以,如下1,原来的Kinematic逻辑,2.切换成自定义代码

整个Component的切换实现,就是利用了反射和Unity Editor的特性

封装了一下代码,可直接调用

//调用方法
//原目標,即使是接口,也可以通过.GetComponent()获取
var currController = go.GetComponent<ICharacterController>();//curr Instance
var motorSource = ReflectionHelper.GetCompField(go,currController.GetType().ToString(),"Motor");
Debug.LogError(motorSource);
static class ReflectionHelper
{
    /// <summary>
    /// 因为UnityEditor原因,这样获取会获取到,Editor库。。。。(如何 ReflectionHelper 类,不放在Editor,也是不会存在获取到Editor库问题)
    /// </summary>
    public static IEnumerable CreateAllInstancesOf<T>()
    {
        return typeof(ReflectionHelper).Assembly.GetTypes() //获取当前类库下所有类型
            //.Where(t => typeof(T).IsAssignableFrom(t)) //获取间接或直接继承t的所有类型
            //.Where(t => !t.IsAbstract && t.IsClass) //获取非抽象类 排除接口继承
            //.Select(t => (T) Activator.CreateInstance(t)); //创造实例,并返回结果(项目需求,可删除)
            .Select(t=>t.FullName);
    }
    
    /// <summary>
    ///  (直接指向,所以能获取Unity Runtime库)
    /// </summary>
    /// <returns></returns>
    public static IEnumerable GetAllClasses<T>()
    {
       // var name = Selection.activeObject.name;//获取 Scene 中 GameObject
        System.Reflection.Assembly assembly = System.Reflection.Assembly.GetExecutingAssembly();
        var dict = System.IO.Path.GetDirectoryName(assembly.Location);
        assembly = System.Reflection.Assembly.LoadFile(System.IO.Path.Combine(dict, "Assembly-CSharp.dll"));
        return assembly.GetTypes()
                .Where(t => typeof(T).IsAssignableFrom(t)) //获取间接或直接继承t的所有类型
                .Where(t => !t.IsAbstract && t.IsClass) //获取非抽象类 排除接口继承
                .Select(t=>t.FullName)
            ;
    }
    /// <summary>
    /// 根据 type 获取go 的 component 的值
    /// </summary>
    /// <param name="go"></param>
    /// <param name="typeString">currController.GetType().ToString()</param>
    /// <param name="name">类的字段 名字</param>
    /// <returns></returns>
    public static object GetCompField(GameObject go, string typeString,string name)
    {
        var assemblies = AppDomain.CurrentDomain.GetAssemblies();
        var defaultAssembly = assemblies.First(assembly => assembly.GetName().Name == "Assembly-CSharp");

        Type typSource = defaultAssembly.GetType(typeString);
        var currInstance = go.GetComponent(typSource);
        FieldInfo fieldSource = typSource.GetField(name);
        return fieldSource.GetValue(currInstance);
        
    }

    public static bool SetCompField(object obj,object objValue, string typeString, string name)
    {
        var assemblies = AppDomain.CurrentDomain.GetAssemblies();
        var defaultAssembly = assemblies.First(assembly => assembly.GetName().Name == "Assembly-CSharp");

        Type typSource = defaultAssembly.GetType(typeString);
        if (typSource == null) return false;
        FieldInfo fieldSource = typSource.GetField(name);
        fieldSource.SetValue(obj,objValue);
        return true;
    }
}

Project View查找的代码实现

void Find(System.Type type)
{
    //step 1:find ref in assets
     
    //filter all GameObject from assets(so-called 'Prefab')
    var guids = AssetDatabase.FindAssets("t:GameObject");
     
    findResult = new List<string>();
     
    var tp = typeof(GameObject);
     
    foreach (var guid in guids)
    {
        var path = AssetDatabase.GUIDToAssetPath(guid);
     
        //load Prefab
        var obj = AssetDatabase.LoadAssetAtPath(path, tp) as GameObject;
     
        //check whether prefab contains script with type 'type'
        if (obj != null)
        {
            var cmp = obj.GetComponent(type);
            if (cmp == null)
            {
                cmp = obj.GetComponentInChildren(type);
            }
            if (cmp != null)
            {
                findResult.Add(path);
            }
        }
    }
     
    //step 2: find ref in scenes
     
    //save current scene
    string curScene = EditorApplication.currentScene;
    EditorApplication.SaveScene();
     
    //find all scenes from dataPath
    string[] scenes = Directory.GetFiles(Application.dataPath, "*.unity", SearchOption.AllDirectories);
     
    //iterates all scenes 
    foreach (var scene in scenes)
    {
        EditorApplication.OpenScene(scene);
     
        //iterates all gameObjects
        foreach (GameObject obj in FindObjectsOfType<GameObject>())
        {
            var cmp = obj.GetComponent(type);
            if (cmp == null)
            {
                cmp = obj.GetComponentInChildren(type);
            }
            if (cmp != null)
            {
                findResult.Add(scene.Substring(Application.dataPath.Length) + "Assets:" + obj.name);
            }
        }
    }
     
    //reopen current scene
    EditorApplication.OpenScene(curScene);
    Debug.Log ("finish");
}

参考:

C#通过反射获取类中的所有字段和属性 - 董川民 (dongchuanmin.com)

Unity-Find-Script-References 查找脚本的引用_子胤的博客-CSDN博客

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/366373.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

人人能读懂redux原理剖析

一、Redux是什么&#xff1f; 众所周知&#xff0c;Redux最早运用于React框架中&#xff0c;是一个全局状态管理器。Redux解决了在开发过程中数据无限层层传递而引发的一系列问题&#xff0c;因此我们有必要来了解一下Redux到底是如何实现的&#xff1f; 二、Redux的核心思想…

计算机网络之IP协议(详解

网络层主管地址管理与路由选择。而IP协议就是网络层中一个非常重要的协议。它的作用就是在复杂的网络环境中确定一个合适的路径。IP协议头格式4位版本号(version) 指定IP协议的版本&#xff0c;目前只有两个版本&#xff1a;IP v4和IP v6.对于IP v4来说&#xff0c;这个值就是4…

边缘云是什么?

涂鸦边缘云服务 旨在解决物联网边缘位置的连接需求和提高设备自主管理能力。并与涂鸦 IoT 云服务和 IoT 终端形成云边端三位一体的端到端产品架构。使用涂鸦边缘云&#xff0c;能极大降低设备响应延时、降低网络带宽压力、提高算力分发能力&#xff0c;并构建以下技术优势&…

IDEA 30 个好用天花板技巧,敲代码直接接爽到飞。

IDEA 作为Java开发工具的后起之秀&#xff0c;几乎以碾压之势把其他对手甩在了身后&#xff0c;主要原因还是归功于&#xff1a;好用&#xff1b;虽然有点重&#xff0c;但依旧瑕不掩瑜&#xff0c;内置了非常多的功能&#xff0c;大大提高了日常的开发效率&#xff0c;下面汇总…

LAMP架构与搭建论坛

目录 1、LAMP架构简述 2、各组件作用 3、构建LAMP平台 1.编译安装Apache httpd服务 2.编译安装mysql 3.编译安装php 4.搭建一个论坛 1、LAMP架构简述 LAMP架构是目前成熟的企业网站应用模式之一&#xff0c;指的是协同工作的一整台系统和相关软件&#xff0c;能够提供动…

Spring Boot整合Thymeleaf和FreeMarker模板

虽然目前市场上多数的开发模式采用前后端分离的技术&#xff0c;视图层的技术在小一些的项目中还是非常有用的&#xff0c;所以一直也占有一席之地&#xff0c;如spring官方的spring.io等网站就是使用视图层技术实现的。 目前Spring Boot支持的较好的两个视图层模板引擎是Thyme…

【git】git版本控制

目录 1.在合适的位置打开bush,创建仓库 2.检查&#xff1a;跳转到当前文件夹&#xff0c;显示当前文件夹的相对路径 3.初始化 4.创建一个文本文件readme.txt 5.手动向readme文件中添加一些内容 6.把文件添加到暂存区 7.把文件提交到git仓库 8.手动修改readme.txt文件 9.查看当前…

前端监控之用户行为监控实践2(数据统计mongodb)

一、技术栈介绍 我们当前的项目&#xff0c;后端是node 搭建&#xff0c;数据库是非关系型数据库 mongodb。 二、数据情况介绍 日志存储存储格式如下&#xff1a; 主要包括&#xff1a; key意义type当前访问类型actionTime访问时间content访问内容erp、fullname、orgname、…

【Spring MVC】这一篇,带你从入门到进阶

目录 1、什么是MVC&#xff1f; 2、什么是 Spring MVC 3、如何学好 Spring MVC&#xff1f; 3.1、如何创建 Spring MVC 项目 3.1.1、使用Spring Initializr创建&#xff08;推荐&#xff09; 3.2、将 Spring 程序与用户&#xff08;浏览器&#xff09;联通 3.3、基础注解…

6.5 拓展:如何实现 Web API 版本控制,同时兼容无版本控制的原始接口?

第6章 构建 RESTful 服务 6.1 RESTful 简介 6.2 构建 RESTful 应用接口 6.3 使用 Swagger 生成 Web API 文档 6.4 实战&#xff1a;实现 Web API 版本控制 6.5 拓展&#xff1a;如何实现 Web API 版本控制&#xff0c;同时兼容无版本控制的原始接口&#xff1f; 6.5 拓展&#…

干旱预测方法总结及基于人工神经网络的干旱预测案例分析(MATLAB全代码)

本案例采用SPEI干旱指数&#xff0c;构建ANN和BP神经网络预测模型&#xff0c;并开展1~3个月预见期的干旱预测&#xff0c;对比分析干旱预测模型的适用性&#xff0c;为流域干旱预警和管理提供技术依据。 干旱预测 1 干旱预测方法 1.1 统计学干旱预测 根据历史降水或气温等…

【python】用plotly绘制正二十面体

文章目录顶点棱实现正二十面体plotly 的 Python 软件包是一个开源的代码库&#xff0c;它基于 plot.js&#xff0c;而后者基于 d3.js。我们实际使用的则是一个对 plotly 进行封装的库&#xff0c;名叫 cufflinks&#xff0c;能让你更方便地使用 plotly 和 Pandas 数据表协同工作…

设备树(配合LED驱动说明)

目录 一、起源 二、基本组成 三、基本语法 四、特殊节点 4.1 根节点 4.2 /memory 4.3 /chosen 4.4 /cpus 多核CPU支持 五、常用属性 5.1 phandle 5.2 地址 --------------- 重要 5.3 compatible --------------- 重要 5.4 中断 --------------- 重要 5.5 …

python攻陷米哈游《元神》数据?详情请看文章。。

前言 嗨喽&#xff0c;大家好呀~这里是爱看美女的茜茜呐 《原神》是由米哈游自研的一款全新开放世界冒险RPG。 里面拥有许多丰富得角色&#xff0c;让玩家为之着迷~ 今天&#xff0c;我们就来用python探索一下原神游戏角色信息&#xff01; 标题大家看看就好了哈~&#xff08…

DNS,DNS污染劫持,DNS加密

1. DNS&#xff08;Domain Name System&#xff09;DNS&#xff08;Domain Name System&#xff09;&#xff0c; 也叫网域名称系统&#xff0c;是互联网的一项服务。它实质上是一个 域名 和 IP 相互映射的分布式数据库.DNS&#xff08;Domain Name Server&#xff0c;域名服务…

医疗保健和智慧城市服务将引领5G物联网采用

Juniper Research预测&#xff0c;到2026年&#xff0c;全球5G物联网连接将达到1.16亿&#xff0c;而2023年仅为1700万。该公司预测&#xff0c;医疗保健部门和智慧城市服务将在未来三年推动这1100%的增长&#xff0c;到2026年占5G物联网设备的60%以上。5G物联网技术的超低延迟…

配置Flutter开发环境

一、在Windows上搭建Flutter开发环境 1、去flutter官网下载其最新可用的安装包&#xff0c;下载地址&#xff1a;https://flutter.dev/docs/development/tools/sdk/releases 。 注意&#xff0c;Flutter的渠道版本一直在不断的更新&#xff0c;请以Flutter官网为准。 另外&…

自动化测试框架对比

Robot Framework&#xff08;RF&#xff09; 链接&#xff1a;http://robotframework.org/ Robot Framework&#xff08;RF&#xff09;是用于验收测试和验收测试驱动开发&#xff08;ATDD&#xff09;的自动化测试框架。 基于 Python 编写&#xff0c;但也可以在 Jython&…

Android 基础知识4-3.1 TextView(文本框)详解

一、前言 TextView就是一个显示文本标签的控件&#xff0c;就是用来显示文本。可以在代码或者 XML中设置字体&#xff0c;字体大小&#xff0c;字体颜色 &#xff0c;字体样式 &#xff08;加粗级斜体&#xff09;&#xff0c;文字截断&#xff08;比如&#xff1a;只显示10个字…

【Python数据挖掘入门】一、数据挖掘概况

一、数据挖掘概况 数据挖掘是指从大量的数据中&#xff0c;通过统计学、人工智能、机器学习等方法&#xff0c;挖掘出未知的、具有价值的信息和知识的过程。 典型案例&#xff1a; 啤酒与尿布杜蕾斯与口香糖杜蕾斯与红酒 数据挖掘是一门交叉学科&#xff0c;覆盖了统计学、数…