Unity3d编辑器扩展学习笔记
编辑器扩展
1.添加菜单栏:把特性应用于静态方法
参数1:菜单名的空格后面是定义快捷键(单符号得用"_"开头,组合键%=Ctrl,#=Shift,&=Alt)
参数2:通过定义一个验证方法来控制是否启用菜单栏(如:当有选择到物体时,启用,否则禁用)
参数3:菜单栏中显示的顺序(优先级),按这个值可以分组(大概相差10就分一组)
可以为已有菜单添加子菜单,如Assets/MyTools,将显示在Assets菜单栏内,同时也会在Projiect窗口的右键菜单内显示。
[MenuItem("MyTools/test1 _t", false, 12)]
[MenuItem("MyTools/test2 %t", false, 12)]
2.利用Selection类的静态方法,可以在编辑器中获取到当前所选择的对象
3.利用Undo类的静态方法,可以执行一些可以撤销的操作
Undo.DestroyObjectImmediate(Selection.activeGameObject);
4.为组件添加右键菜单栏
方式1:可以为自定义组件、内置组件添加
必须以“CONTEXT”开始,方法的参数MenuCommand mc会被自动赋值为该组件所挂载的游戏物体
[MenuItem("CONTEXT/PlayerController/菜单名")]
static void ResetSize(MenuCommand mc)
{
PlayerController pc = mc.context as PlayerController;
//接下来就可以修改组件对象pc的属性了
pc.Size = 12;
}
方式2:只能为自定义组件添加(可以编辑代码的组件),直接在该组件的代码内添加。
ContextMenu和ContextMenuItem类在UnityEngine命名空间下,与UnityEditor无关
a.为组件添加右键菜单栏
[ContextMenu("Set Color")]
void SetColor()
{
bodyColor = Color.red;
}
b.为组件的属性添加右键菜单栏
[ContextMenuItem("Menu name", "AddSize")]
public int Size = 0;
void AddSize()
{
Size += 10;
}
5.自定义Inspector面板
using UnityEngine;
using UnityEditor;
using System.Collections.Generic;
// 为组件EnemyManager自定义Inspector面板显示,CanEditorMutipleObjects 表示可以同时编辑多个物体
[CustomEditor(typeof(EnemyManager)),CanEditMultipleObjects]
public class EnemyManagerInspector : Editor
{
// 这里用简单的显示控制来让大家理解自定义编辑器的流程
public static SerializedProperty enemyList, bossEnemyList,size;
void OnEnable()
{
enemyList = serializedObject.FindProperty("EnemyList");
bossEnemyList = serializedObject.FindProperty("BossEnemyList");
}
private bool isOpenDelete = false;
private bool isOpenAdd = true;
private bool isShow = false;
public override void OnInspectorGUI()
{
//表示只做附加显示,可为内置组件附加新的Inspector面板属性
base.DrawDefaultInspector();
// 更新编辑器显示的序列化属性
serializedObject.Update();
//获取组件对象
EnemyManager enemyManager = (EnemyManager)target;
//显示属性,true表示显示所有包含的成员
EditorGUILayout.PropertyField(enemyList,true);
//开始新的一行
EditorGUILayout.BeginHorizontal();
isOpenDelete = EditorGUILayout.BeginToggleGroup("是否允许删除", isOpenDelete); //包含一组元素,整体控制他们是否可编辑
if (GUILayout.Button("删除成员"))
{
}
EditorGUILayout.EndToggleGroup();
EditorGUILayout.EndHorizontal();
GUILayout.Label("**********分割线*********");
// 接受序列化赋值
serializedObject.ApplyModifiedProperties();
}
}
6.自定义窗口
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEditor;
public class MyWindow : EditorWindow {
[MenuItem("Window/show mywindow")]
static void ShowMyWindow()
{
MyWindow window= EditorWindow.GetWindow<MyWindow>();
window.Show();
}
private string name="";
void OnGUI()
{
GUILayout.Label("这是我的窗口");
name = GUILayout.TextField(name);
if (GUILayout.Button("创建"))
{
GameObject go = new GameObject(name);
//纪录操作,便于撤销
Undo.RegisterCreatedObjectUndo(go, "create gameobject");
}
}
}
7.自定义对话窗口
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEditor;
public class EnemyChange : ScriptableWizard {
[MenuItem("Tools/CreateWizard")]
static void CreateWizard()
{
ScriptableWizard.DisplayWizard<EnemyChange>("统一修改敌人","Change And Close","Change");
}
public int changeStartHealthValue = 10;
public int changeSinkSpeedValue = 1;
const string changeStartHealthValueKey = "EnemyChange.changeStartHealthValue";
const string changeSinkSpeedValueKey = "EnemyChange.changeSinkSpeedValue";
//当窗口被创建出来的时候调用的
void OnEnable()
{
changeStartHealthValue = EditorPrefs.GetInt(changeStartHealthValueKey, changeStartHealthValue);
changeSinkSpeedValue = EditorPrefs.GetInt(changeSinkSpeedValueKey, changeSinkSpeedValue);
}
//检测create按钮的点击
void OnWizardCreate()
{
GameObject[] enemyPrefabs = Selection.gameObjects; //获得选择的物体对象
//显示进度条
EditorUtility.DisplayProgressBar("进度", "0/" + enemyPrefabs.Length + " 完成修改值", 0);
int count = 0;
foreach (GameObject go in enemyPrefabs)
{
CompleteProject.EnemyHealth hp = go.GetComponent<CompleteProject.EnemyHealth>();
Undo.RecordObject(hp, "change health and speed");
hp.startingHealth += changeStartHealthValue;
hp.sinkSpeed += changeSinkSpeedValue;
count++;
EditorUtility.DisplayProgressBar("进度", count+"/" + enemyPrefabs.Length + " 完成修改值", (float)count/enemyPrefabs.Length);
}
EditorUtility.ClearProgressBar();
//显示提示信息
ShowNotification(new GUIContent(Selection.gameObjects.Length + "个游戏物体的值被修改了"));
}
//检测Other按钮的点击
void OnWizardOtherButton()
{
OnWizardCreate();
}
//当前字段值修改的时候会被调用
void OnWizardUpdate()
{
errorString = null;
helpString = null;
if (Selection.gameObjects.Length > 0)
{
helpString = "您当前选择了" + Selection.gameObjects.Length + "个敌人"; //头部提示信息
}
else
{
errorString = "请选择至少一个敌人"; //底部警告信息
}
//保存面板的修改
EditorPrefs.SetInt(changeStartHealthValueKey, changeStartHealthValue);
EditorPrefs.SetInt(changeSinkSpeedValueKey, changeSinkSpeedValue);
}
void OnSelectionChange()
{
OnWizardUpdate();
}
}
8.其它
当脚本挂载到物体上时,自动为它添加指定组件,如:[RequireComponent(typeof(Rigidbody))]
添加进菜单Component中,[AddComponentMenu("companentPathAndName")]
Unity3d编辑器扩展学习笔记的更多相关文章
- Unity3D编辑器扩展(六)——模态窗口
前面我们已经写了5篇关于编辑器的,这是第六篇,也是最后一篇: Unity3D编辑器扩展(一)——定义自己的菜单按钮 Unity3D编辑器扩展(二)——定义自己的窗口 Unity3D编辑器扩展(三)—— ...
- Unity3D编辑器扩展(五)——常用特性(Attribute)以及Selection类
前面写了四篇关于编辑器的: Unity3D编辑器扩展(一)——定义自己的菜单按钮 Unity3D编辑器扩展(二)——定义自己的窗口 Unity3D编辑器扩展(三)——使用GUI绘制窗口 Unity3D ...
- Unity3D编辑器扩展(四)——扩展自己的组件
前面已经写了三篇: Unity3D编辑器扩展(一)——定义自己的菜单按钮 Unity3D编辑器扩展(二)——定义自己的窗口 Unity3D编辑器扩展(三)——使用GUI绘制窗口 今天写第四篇,扩展自己 ...
- Unity3D之UGUI学习笔记(一):UGUI介绍以及Canvas
UGUI是Unity3D4.6官方提供的UI系统,支持2D和3D UI的开发. Unity3D UI史 OnGUI 在Unity4.6之前,官方提供的是OnGUI函数来开发UI界面,当然问题也比较多, ...
- Unity3D之UGUI学习笔记(三):EventSystem
在UGUI中,EventSystem实现了所有关于交互方面的功能,和NGUI不一样的地方是,我们终于可以摆脱添加Box Collider了! 下面我们来学习一下. 对于按钮来说,直接有onClick的 ...
- unity3D编辑器扩展
编辑器扩展只是在编辑项目中运行,发布出来是不会运行的. 固定创建一个文件夹Editor:所有的资源或者代码都不会被打包进去. 01.使用MenuItem添加菜单栏按钮 脚本不需要作为组件存在,可以不用 ...
- 【第五课】VIM编辑器(学习笔记)
4月10日学习笔记打卡
- Unity3D编辑器扩展(一)——定义自己的菜单按钮
Unity3D 引擎的编辑器拥有很强的扩展性,用的好可以帮我们省很多事情.在这里记录下如何去扩展 Unity3D 的编辑器,定制属于我们自己的开发环境. 本篇主要讲解在 Unity3D 引擎的各个窗口 ...
- Unity3D之ScriptableObject学习笔记
不同与C#提供的Serializable序列化功能,ScriptableObject是Unity3D提供的一个数据存储类,我们接下来学习一下这个类的功能. 官方文档 http://docs.unity ...
随机推荐
- 转:javascript判断IE浏览器
http://blog.csdn.net/ranbolwb/article/details/18555847 function isIE() { //ie? if (!!window.ActiveXO ...
- django DatabaseFunctions
from django.db.functions import ... Cast() 转换类型 value = Value.objects.annotate(field_as_float=Cast(' ...
- RAID 介绍
介绍 磁盘阵列(Redundant Arrays of Independent Disks,RAID),有“独立磁盘构成的具有冗余能力的阵列”之意. 磁盘阵列是由很多价格较便宜的磁盘,组合成一个容量巨 ...
- 「HNOI 2013」消毒
题目链接 戳我 \(Solution\) 我们首先想一想如果这一题只是二维的该怎么办? 就是一个最小点覆盖问题.这里就不详细解释了,用网络流或匈牙利都无所谓. 但现在是三维的,那么现在该如何处理呢? ...
- pandas set_index() reset_index()
set_index() 官方定义: 使用一个或多个现有列设置索引, 默认情况下生成一个新对象 DataFrame.set_index(keys, drop=True, append=False, ...
- gulp 图片、样式、js、实时刷新等压缩gulpfile.js文件各个模块
1.压缩tinypng图片 gulp-tinypng-nokey,但不压缩gif格式(另外一个gulp-imagemin压缩率不高,可以压缩gif格式) // 获取 gulp var gulp = ...
- Oracle TNS Listener Remote Poisoning
Oracle TNS Listener Remote Poisoning 远程数据投毒漏洞(CVE-2012-1675) 1.漏洞简介: 允许攻击者在不提供用户名/密码的情况下,向远程“TNS Lis ...
- 搭建sftp并设置不同权限的多个用户
一, 设置相关用户,用户组,ssh配置文件 mkdir -pv /opt/ftpsite/{admin,user1,user2} groupadd sftpadmins groupadd sftpus ...
- mysql5.1解压版安装
1.如果已经安装别的版本先卸载干净,cmd管理员权限登录,mysql\bin目录:mysqld -remove 2.环境变量 MYSQL:path 3.my.ini [mysqld]port=3306 ...
- Vue页面加载时,触发某个函数的方法
需要在加载页面的时候调用生成验证码的click事件函数 解决方法如下,利用Vue中的mounted mounted:function(){ this.createcode();//需要触发的函数 } ...