Unity 编辑器开发实战【Editor Window】- Filter 物体筛选工具

简介: Unity 编辑器开发实战【Editor Window】- Filter 物体筛选工具

Unity开发工作中,在Hierarchy窗口搜索栏可以通过物体名称或组件名称对场景中的物体进行搜索,但是并不能满足我们一些其它的搜索要求,例如搜索指定Tag标签的物体,或者指定Layer层级的物体,或者指定Active状态的物体,或者更为复杂的一些搜索,比如我们想找到场景中所有隐藏的、且挂有Camera组件的、且标签为MainCamera的物体,这些都无法实现。

今天分享一个作者为了解决上述搜索需求而开发的Filter物体筛选器:

image.gif

其中Target是指需要进行筛选的所有物体,All是指对场景中的所有物体进行筛选,也可以指定一个根级,对这个根物体的所有子物体进行筛选:

image.gif

确定好要进行筛选的物体后,下面来创建筛选条件:

image.gif

1.Name 通过物体名称的关键字进行筛选

image.gif

2.Component 通过组件进行筛选 -物体是否挂有指定组件

image.gif

3.Layer 通过物体的Layer层级进行筛选

image.gif

4.Tag 通过物体的Tag标签进行筛选

image.gif

5.Active 通过物体的活跃状态进行筛选

image.gif

以上是单个条件的筛选方式,我们也可以创建复合条件,即多个条件对物体进行筛选,比如文章开始提到的,我们要找到场景中所有隐藏的、且挂有Camera组件的、且标签为MainCamera的物体,需要创建3个条件:1.Active活跃状态为false条件、2.Component组件为Camera条件、3.Tag标签为MainCamera条件

image.gif

最终点击Select按钮可以选中全部我们筛选出的符合条件的物体,以下是实现代码:

usingUnityEngine;
usingUnityEditor;
usingUnityEngine.SceneManagement;
usingSystem;
usingSystem.Reflection;
usingSystem.Collections.Generic;
namespaceSK.Framework{
/// <summary>/// 过滤类型/// </summary>publicenumFilterMode    {
Name, //根据名字筛选Component, //根据组件筛选Layer, //根据层级筛选Tag, //根据标签筛选Active, //根据是否活跃筛选Missing, //丢失筛选    }
publicenumMissingMode    {
Material, //材质丢失Mesh, //网格丢失Script//脚本丢失    }
    [SerializeField]
publicclassFilterCondition    {
publicFilterModefilterMode;
publicMissingModemissingMode;
publicstringstringValue;
publicintintValue;
publicboolboolValue;
publicTypetypeValue;
publicFilterCondition(FilterModefilterMode, stringstringValue)
        {
this.filterMode=filterMode;
this.stringValue=stringValue;
        }
publicFilterCondition(FilterModefilterMode, intintValue)
        {
this.filterMode=filterMode;
this.intValue=intValue;
        }
publicFilterCondition(FilterModefilterMode, boolboolValue)
        {
this.filterMode=filterMode;
this.boolValue=boolValue;
        }
publicFilterCondition(FilterModefilterMode, TypetypeValue)
        {
this.filterMode=filterMode;
this.typeValue=typeValue;
        }
publicFilterCondition(FilterModefilterMode, MissingModemissingMode)
        {
this.filterMode=filterMode;
this.missingMode=missingMode;
        }
/// <summary>/// 判断物体是否符合条件/// </summary>/// <param name="target">物体</param>/// <returns>符合条件返回true,否则返回false</returns>publicboolIsMatch(GameObjecttarget)
        {
switch (filterMode)
            {
caseFilterMode.Name: returntarget.name.ToLower().Contains(stringValue.ToLower());
caseFilterMode.Component: returntarget.GetComponent(typeValue) !=null;
caseFilterMode.Layer: returntarget.layer==intValue;
caseFilterMode.Tag: returntarget.CompareTag(stringValue);
caseFilterMode.Active: returntarget.activeSelf==boolValue;
caseFilterMode.Missing:
switch (missingMode)
                    {
caseMissingMode.Material:
varmr=target.GetComponent<MeshRenderer>();
if (mr==null) returnfalse;
Material[] materials=mr.sharedMaterials;
boolflag=false;
for (inti=0; i<materials.Length; i++)
                            {
if(materials[i] ==null)
                                {
flag=true;
break;
                                }
                            }
returnflag;
caseMissingMode.Mesh:
varmf=target.GetComponent<MeshFilter>();
if (mf==null) returnfalse;
returnmf.sharedMesh==null;
caseMissingMode.Script:
Component[] components=target.GetComponents<Component>();
boolretV=false; 
for (inti=0; i<components.Length; i++)
                            {
if(components[i] ==null)
                                {
retV=true;
break;
                                }
                            }
returnretV;
default:
returnfalse;
                    }
default: returnfalse;
            }
        }
    }
publicsealedclassFilter : EditorWindow    {
        [MenuItem("SKFramework/Tools/Filter")]
privatestaticvoidOpen()
        {
varwindow=GetWindow<Filter>();
window.titleContent=newGUIContent("Filter");
window.Show();
        }
//筛选的目标privateenumFilterTarget        {
All, //在所有物体中筛选Specified, //在指定根级物体内筛选        }
privateFilterTargetfilterTarget=FilterTarget.All;
//存储所有筛选条件privatereadonlyList<FilterCondition>filterConditions=newList<FilterCondition>();
//指定的筛选根级privateTransformspecifiedTarget;
//存储所有组件类型privateList<Type>components;
//存储所有组件名称privateList<string>componentsNames;
privatereadonlyList<GameObject>selectedObjects=newList<GameObject>();
privateVector2scroll=Vector2.zero;
privatevoidOnEnable()
        {
components=newList<Type>();
componentsNames=newList<string>();
Assembly[] assemblies=AppDomain.CurrentDomain.GetAssemblies();
for (inti=0; i<assemblies.Length; i++)
            {
vartypes=assemblies[i].GetTypes();
for (intj=0; j<types.Length; j++)
                {
if (types[j].IsSubclassOf(typeof(Component)))
                    {
components.Add(types[j]);
componentsNames.Add(types[j].Name);
                    }
                }
            }
        }
privatevoidOnGUI()
        {
OnTargetGUI();
scroll=EditorGUILayout.BeginScrollView(scroll);
OnConditionGUI();
OnIsMatchedGameObjectsGUI();
EditorGUILayout.EndScrollView();
OnFilterGUI();
        }
privatevoidOnTargetGUI()
        {
filterTarget= (FilterTarget)EditorGUILayout.EnumPopup("Target", filterTarget);
switch (filterTarget)
            {
caseFilterTarget.Specified:
specifiedTarget=EditorGUILayout.ObjectField("Root", specifiedTarget, typeof(Transform), true) asTransform;
break;
            }
EditorGUILayout.Space();
        }
privatevoidOnConditionGUI()
        {
if (GUILayout.Button("Create New Condition", "DropDownButton"))
            {
GenericMenugm=newGenericMenu();
gm.AddItem(newGUIContent("Name"), false, () =>filterConditions.Add(newFilterCondition(FilterMode.Name, "GameObject")));
gm.AddItem(newGUIContent("Component"), false, () =>filterConditions.Add(newFilterCondition(FilterMode.Component, typeof(Transform))));
gm.AddItem(newGUIContent("Layer"), false, () =>filterConditions.Add(newFilterCondition(FilterMode.Layer, 0)));
gm.AddItem(newGUIContent("Tag"), false, () =>filterConditions.Add(newFilterCondition(FilterMode.Tag, "Untagged")));
gm.AddItem(newGUIContent("Active"), false, () =>filterConditions.Add(newFilterCondition(FilterMode.Active, true)));
gm.AddItem(newGUIContent("Missing / Material"), false, () =>filterConditions.Add(newFilterCondition(FilterMode.Missing, MissingMode.Material)));
gm.AddItem(newGUIContent("Missing / Mesh"), false, () =>filterConditions.Add(newFilterCondition(FilterMode.Missing, MissingMode.Mesh)));
gm.AddItem(newGUIContent("Missing / Script"), false, () =>filterConditions.Add(newFilterCondition(FilterMode.Missing, MissingMode.Script)));
gm.ShowAsContext();
            }
EditorGUILayout.Space();
if(filterConditions.Count>0)
            {
GUILayout.BeginVertical("Badge");
for (inti=0; i<filterConditions.Count; i++)
                {
varcondition=filterConditions[i];
GUILayout.BeginHorizontal();
if(filterConditions.Count>1)
GUILayout.Label($"{i + 1}.", GUILayout.Width(30f));
switch (condition.filterMode)
                    {
caseFilterMode.Name:
GUILayout.Label("Name", GUILayout.Width(80f));
condition.stringValue=EditorGUILayout.TextField(condition.stringValue);
break;
caseFilterMode.Component:
varindex=componentsNames.FindIndex(m=>m==condition.typeValue.Name);
GUILayout.Label("Component", GUILayout.Width(80f));
varnewIndex=EditorGUILayout.Popup(index, componentsNames.ToArray());
if (index!=newIndex) condition.typeValue=components[newIndex];
break;
caseFilterMode.Layer:
GUILayout.Label("Layer", GUILayout.Width(80f));
condition.intValue=EditorGUILayout.LayerField(condition.intValue);
break;
caseFilterMode.Tag:
GUILayout.Label("Tag", GUILayout.Width(80f));
condition.stringValue=EditorGUILayout.TagField(condition.stringValue);
break;
caseFilterMode.Active:
GUILayout.Label("Active", GUILayout.Width(80f));
condition.boolValue=EditorGUILayout.Toggle(condition.boolValue);
break;
caseFilterMode.Missing:
GUILayout.Label("Missing", GUILayout.Width(80f));
condition.missingMode= (MissingMode)EditorGUILayout.EnumPopup(condition.missingMode);
break;
default:
break;
                    }
if (GUILayout.Button("×", "MiniButton", GUILayout.Width(20f)))
                    {
filterConditions.RemoveAt(i);
return;
                    }
GUILayout.EndHorizontal();
                }
GUILayout.EndVertical();
            }
EditorGUILayout.Space();
        }
privatevoidOnIsMatchedGameObjectsGUI()
        {
for (inti=0; i<selectedObjects.Count; i++)
            {
GameObjectobj=selectedObjects[i];
if(obj==null)
                {
selectedObjects.RemoveAt(i);
i--;
continue;
                }
GUILayout.BeginHorizontal("IN Title");
GUILayout.Label(obj.name);
GUILayout.EndHorizontal();
if (Event.current.type==EventType.MouseDown&&GUILayoutUtility.GetLastRect().Contains(Event.current.mousePosition))
                {
EditorGUIUtility.PingObject(obj);
                }
            }
GUILayout.FlexibleSpace();
        }
privatevoidOnFilterGUI()
        {
GUILayout.BeginHorizontal();
if (GUILayout.Button("Filter", "ButtonLeft"))
            {
selectedObjects.Clear();
List<GameObject>targetGameObjects=newList<GameObject>();
switch (filterTarget)
                {
caseFilterTarget.All:
GameObject[] rootGameObjects=SceneManager.GetActiveScene().GetRootGameObjects();
for (inti=0; i<rootGameObjects.Length; i++)
                        {
varroot=rootGameObjects[i];
varallChildren=root.GetComponentsInChildren<Transform>(true);
for (intj=0; j<allChildren.Length; j++)
                            {
EditorUtility.DisplayProgressBar("Filter", allChildren[j].name, (float)i/rootGameObjects.Length);
targetGameObjects.Add(allChildren[j].gameObject);
                            }
                        }
EditorUtility.ClearProgressBar();
break;
caseFilterTarget.Specified:
Transform[] children=specifiedTarget.GetComponentsInChildren<Transform>(true);
for (inti=0; i<children.Length; i++)
                        {
EditorUtility.DisplayProgressBar("Filter", children[i].name, (float)i/children.Length);
targetGameObjects.Add(children[i].gameObject);
                        }
EditorUtility.ClearProgressBar();
break;
default:
break;
                }
for (inti=0; i<targetGameObjects.Count; i++)
                {
GameObjecttarget=targetGameObjects[i];
boolisMatch=true;
for (intj=0; j<filterConditions.Count; j++)
                    {
if (!filterConditions[j].IsMatch(target))
                        {
isMatch=false;
break;
                        }
                    }
EditorUtility.DisplayProgressBar("Filter", $"{target.name} -> Is Matched : {isMatch}", (float)i/targetGameObjects.Count);
if (isMatch)
                    {
selectedObjects.Add(target);
                    }
                }
EditorUtility.ClearProgressBar();
            }
if (GUILayout.Button("Select", "ButtonMid"))
            {
Selection.objects=selectedObjects.ToArray();
            }
if (GUILayout.Button("Clear", "ButtonRight"))
            {
selectedObjects.Clear();
            }
GUILayout.EndHorizontal();
        }
    }
}

image.gif

目录
相关文章
|
3月前
|
C# 图形学 开发者
Unity开发中使用UnityWebRequest从HTTP服务器下载资源。
总之,UnityWebRequest就是游戏开发者手中的万能钓鱼竿,既可以获取文本数据,也能钓上图片资源,甚至是那声音的涟漪。使用UnityWebRequest的时候,你需要精心准备,比如确定URL、配置请求类型和头信息;发起请求;巧妙处理钓获的数据;还需要机智面对网络波澜,处理各种可能出现的错误。按照这样的过程,数据的钓取将会是一次既轻松愉快也效率高效的编程钓鱼之旅。
183 18
|
8月前
|
开发框架 Java 编译器
2025年1月推荐-工欲善其事,必先利其器-程序员必备之-核心基本工具—不要看什么国际排行榜-没有用-编辑器和编译器推荐-优雅草央千澈
2025年1月推荐-工欲善其事,必先利其器-程序员必备之-核心基本工具—不要看什么国际排行榜-没有用-编辑器和编译器推荐-优雅草央千澈
287 1
|
7月前
|
图形学
Unity 获取鼠标位置下的UGUI或3D物体
本文总结了两种检测方法,分别用于UGUI和3D物体的检测。第一种方法`GetOverUIobj`专门用于检测鼠标悬停的UGUI元素,通过`GraphicRaycaster`实现。第二种方法`GetOverWordGameObject`则同时适用于UI和3D物体检测,利用`PhysicsRaycaster`进行射线检测。两者均返回悬停对象或null。
|
7月前
|
图形学
unity 物体震动
在Unity中实现物体震动效果,主要通过改变物体的位置、旋转或缩放属性来模拟震动。以下是位置震动的实现原理及代码示例:通过随机生成微小偏移量并累加到物体位置上,在短时间内不断改变位置产生震动效果。生成随机偏移,并结合时间控制持续震动。
|
7月前
|
前端开发 图形学
unity UGUI跟随3D物体的坐标转换
在 Unity 中实现 UGUI 元素跟随 3D 物体,关键是将 3D 物体的世界坐标转换为屏幕或画布坐标。通过 Camera.WorldToScreenPoint 方法,可将 3D 物体位置映射到屏幕上,再更新 UGUI 元素的位置。代码示例展示了如何使用该方法,使 UGUI 图像跟随 3D 模型,并提供文字显示、图像和线条的显示/隐藏功能。
|
7月前
|
存储 图形学 索引
unity 使物体跟随路径点自动移动位置
在Unity中,物体沿路径点自动移动的核心原理是通过预设路径点,控制物体依次移动。路径点可用空对象或三维向量数组定义,并按顺序存储。移动时,计算当前位置与下一个路径点的向量差以确定方向,使用`Vector3.MoveTowards`逐步靠近目标点。代码实现包括路径点设置、移动控制及插值计算,确保物体平滑移动和旋转。
|
7月前
|
图形学
Unity 射线移动物体Ray
在Unity中,通过射线检测实现3D物体的拖拽和移动。射线由起点和方向组成,使用`Physics.Raycast`检测与物体的交点。点击物体时,记录位置偏移量,拖动过程中更新物体位置。代码包括基本拖拽和上下拖动功能,适用于正交摄像机场景。测试时为物体设置特定标签(如&quot;JQR&quot;)以便区分和操作。 示例代码展示了如何通过鼠标事件控制物体移动,并结合层级掩码优化射线检测。具体实现包括:点击选中物体、拖动更新位置、释放鼠标取消选择。此外,提供了上下拖动的额外功能,通过按键切换模式。
|
7月前
|
图形学 开发者
unity 从工具栏拖动生成物体
在 Unity 中实现从工具栏拖动生成物体的功能,基于编辑器扩展、事件系统和预制体实例化。通过自定义编辑器窗口、处理鼠标事件(按下、移动、释放)及使用 Instantiate 方法,可实现拖动并生成预制体物体。代码示例展示了如何检测鼠标事件并在指定位置实例化物体。
|
9月前
|
存储 人工智能 数据库
Codel:AI代理工具,支持在终端、浏览器、编辑器执行复杂任务和项目
Codel是一款全自主AI代理工具,支持在终端、浏览器和编辑器中执行复杂任务和项目。它运行在沙盒化的Docker环境中,具备自主操作能力,内置浏览器和文本编辑器,所有操作记录存储于PostgreSQL数据库。Codel能够自动完成复杂任务,如创建项目结构、进行网络搜索等,适用于自动化编程、研究与开发、教育与培训以及数据科学与分析等多个领域。
270 11
Codel:AI代理工具,支持在终端、浏览器、编辑器执行复杂任务和项目
|
10月前
|
前端开发
业余时间开发了个海报编辑器
为了满足撰写博客或录制教程视频时对高质量海报的需求,我利用业余时间开发了一款海报编辑器。第一版功能简单,支持固定尺寸、黑底白字的标题。后来经过优化,增加了背景图、模糊效果、文字样式调整等功能,使海报更具吸引力。目前该编辑器已上线,欢迎大家试用并反馈。[访问海报编辑器](https://tool.share888.top/#/poster)
166 6
业余时间开发了个海报编辑器