Unity插件扩展中组件常用的几个方法
最近为美术编写一个Unity编辑器的扩展,主要为了减轻美术在修改预制对象时的机械化操作的繁琐和出错。具体实现的几个功能:
1、删除指定组件;
2、复制、粘贴指定的组件;
3、重新关联新的属性;
4、重新保存预制对象;
一、删除指定类型的组件
public static void RemoveComponentHandler(GameObject gameObject, Type componentType)
{
foreach (var component in gameObject.GetComponents<Component>())
{
if (component.GetType() == componentType)
{
GameObject.DestroyImmediate(component);
}
}
}
二、复制组件(这里实现的是一次仅复制一个某类型的组件)
public static void CopyComponentHandler(Type componentType, GameObject fromGameObject, GameObject toGameObject)
{
RemoveComponentHandler(toGameObject, componentType); // 查找需要复制的 Component
Component needCopyComponent = null;
foreach (var component in fromGameObject.GetComponents<Component>())
{
if (component.GetType() == componentType)
{
needCopyComponent = component;
break;
}
} // 进行粘贴操作
// http://answers.unity3d.com/questions/907294/copy-all-components-from-a-gameobject-and-paste-to.html
UnityEditorInternal.ComponentUtility.CopyComponent(needCopyComponent);
UnityEditorInternal.ComponentUtility.PasteComponentAsNew(toGameObject);
}
三、关联新属性
就是遍历指定的GameObject,然后找到它附加的组件,重新设置其值即可。
四、替换预制对象
GameObject activeGameObject = Selection.activeGameObject;
if (activeGameObject != null)
{
// 获取当前的id
if (new Regex(@"^\d+h$").IsMatch(activeGameObject.name))
{
UnityEngine.Object parentObject = null;
string strPrefabPath = ""; if (PrefabUtility.GetPrefabType(activeGameObject) == PrefabType.PrefabInstance)
{
parentObject = EditorUtility.GetPrefabParent(activeGameObject);
strPrefabPath = AssetDatabase.GetAssetPath(parentObject);
} // 查找id
string strId = new Regex(@"h$").Replace(activeGameObject.name, ""); // 第六步 保存预制对象
string strCurrSelectPrefabName = activeGameObject.name;
if (strPrefabPath.EndsWith(".prefab"))
{
// string[] dependPaths = AssetDatabase.GetDependencies(strPrefabPath);
GameObject go = GameObject.Instantiate(GameObject.Find(strCurrSelectPrefabName)) as GameObject;
PrefabUtility.ReplacePrefab(go, parentObject, ReplacePrefabOptions.ConnectToPrefab); GameObject.DestroyImmediate(activeGameObject);
go.name = strCurrSelectPrefabName; AssetDatabase.Refresh();
} Debug.Log("预制对象 " + strCurrSelectPrefabName + " 修改完成。"); }
else
{
Debug.Log("当前选中的GameObject命名不符合要求,格式:id+h。\tGameObject Name : " + activeGameObject.name);
}
}
最核心的几行代码:
1、实例化一个新的GameObject;
2、替换预制对象;
3、销毁老的GameObject;
4、刷新资源;
对于美术的同事来讲,最复杂、麻烦的莫过于重新关联属性,特别是骨骼动画。因为之前没有统一的规范,所以关联哪一段动画实际上是需要一层一层找的,我看着他们找都觉得累,怎么办呢?我想到一个办法,就是通过name查找新的组件,然后重新赋值关联。通过Name查找某个GameObject下的子节点(前提条件是该Name唯一)
public static GameObject FindChildGameObject(GameObject parent, string childName)
{
if (parent.name == childName)
{
return parent;
} if (parent.transform.childCount < 1)
{
return null;
} GameObject obj = null;
for (int i = 0; i < parent.transform.childCount; i++)
{
GameObject go = parent.transform.GetChild(i).gameObject;
obj = FindChildGameObject(go, childName);
if (obj != null)
{
break;
}
}
return obj;
}
上面基本上实现了,组件几个常用的方法:
1、添加组件(先复制后粘贴);
2、删除组件;
3、通过名字查找子组件;
4、更新预制对象;
Unity插件扩展中组件常用的几个方法的更多相关文章
- [Unity]Unity3D编辑器插件扩展和组件扩展
1. 插件扩展 1.1. 命名空间 using UnityEditor; using UnityEngine; //非必需,常用到 1.2. 使用语法 [MenuItem("Assets/M ...
- Vue中组件通信的几种方法(Vue3的7种和Vue2的12种组件通信)
Vue3组件通信方式: props $emit expose / ref $attrs v-model provide / inject Vuex 使用方法: props 用 props 传数据给子组 ...
- 通过写一个Demo展示C#中多种常用的集合排序方法
不多说,程序很简单,就是将集合中的数据进行排序,但使用到的知识点还是比较多的,大牛勿喷,谨献给初学者!直接上程序吧! namespace Demo { /// <summary> /// ...
- JAVA中正则表达式常用的四个方法
JAVA中正则表达式处理字符串的四个常用方法:匹配.分割.替换.截取.其跟字符串的常用函数相似,但是使用正则表达式会更简单.更加简洁.下面是具体的例子: public class TestRegex ...
- laravel中及其常用的一些函数方法(自己看)和技巧(不断添加中)
手册:https://laravelacademy.org/ 1.中间件的定义Middleware 2.路由的定义和写法 3.控制器Controller之Request 4.控制器Controller ...
- db2 查看进程 db2中的常用命令及使用方法
一 高(重要度) 1 启动一个db 2实例使用: net start instanceName 2 停止一个db 2实例使用: net stop instanceName 3 启动配置助手: db2= ...
- Hutool中那些常用的工具类和方法
Hutool中那些常用的工具类和方法 Hutool是一个Java工具包,它帮助我们简化每一行代码,避免重复造轮子.如果你有需要用到某些工具方法的时候,不妨在Hutool里面找找,可能就有.本文将对Hu ...
- nginx 常用的 URL 重写方法
转自:http://www.jbxue.com/article/4727.html Nginx中一些常用的URL 重写方法介绍,有需要的朋友可以参考下.url重写应该不陌生,不管是SEO URL 伪静 ...
- JavaScript中的Array.prototype.slice.call()方法学习
JavaScript中的Array.prototype.slice.call(arguments)能将有length属性的对象转换为数组(特别注意: 这个对象一定要有length属性). 但有一个例外 ...
随机推荐
- flink的集群的HA高可用
对于一个企业级的应用,稳定性是首要要考虑的问题,然后才是性能,因此 HA 机制是必不可少的: 和 Hadoop 一代一样,从架构中我们可以很明显的发现 JobManager 有明显的单点问题(SPOF ...
- Spring AOP中args()、arg-names、argNames
先小结一下: args()是用来匹配并且接收目标方法的参数的. argNames(用在注解中)与arg-names(用在XML中),他们是同一个东西. argNames用来接收AspectJ表达式中的 ...
- 在vue-cli项目中使用bootstrap的方法示例
在一个html页面中加入bootstrap是很方便,就是一般的将css和js文件通过Link和Script标签就行. 那么在一个用vue-cli生成的前端项目中如何加入?因为框架不一样了,略微要适应一 ...
- Ubuntu18.04上安装Docker CE
建立 REPOSITORY 1.更新索引包 更新 /etc/apt/sources.list 和 /etc/apt/sources.list.d 中列出的源的地址,这样才能获取到最新的软件包 sudo ...
- Python交互图表可视化Bokeh:3. 散点图
散点图 ① 基本散点图绘制② 散点图颜色.大小设置方法③ 不同符号的散点图 1. 基本散点图绘制 import numpy as np import pandas as pd import matpl ...
- 20165235 实验三 敏捷开发与XP实践
20165235 实验三 敏捷开发与XP实践 主目录: 姓名:祁瑛 学号:20165235 班级:1652 实验课程:JAVA程序设计 实验名称:Java面向对象程序设计 实验时间:2018.4.30 ...
- Jupyter运行时出现下面的错误:Unexpected error while saving file: arma/Untitled.ipynb [Errno 13] Permission denied:
运行环境:Ubuntu16.04+Python2.7执行如下代码修改Jupyter的一部分文件的权限(执行完之后重新启动即可): sudo chmod ~/.local/share/jupyter/ ...
- 问题 L: An Invisible Hand - (2018年第二阶段个人训练赛第三场)
题目描述 There are N towns located in a line, conveniently numbered 1 through N. Takahashi the merchant ...
- 20172328《程序设计与数据结构》实验四 Android程序设计报告
20172328<程序设计与数据结构>实验四 Android程序设计报告 课程:<程序设计与数据结构> 班级: 1723 姓名: 李馨雨 学号:20172328 实验教师:王志 ...
- ORA-01000: maximum open cursors exceeded
网上搜索了一下,找到了原因根源: 使用Oracle数据库的时候,经常会碰到有ORA-01000: maximum open cursors exceeded的错误.实际上,这个错误的原因,主要还是代码 ...