编辑器扩展

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编辑器扩展学习笔记的更多相关文章

  1. Unity3D编辑器扩展(六)——模态窗口

    前面我们已经写了5篇关于编辑器的,这是第六篇,也是最后一篇: Unity3D编辑器扩展(一)——定义自己的菜单按钮 Unity3D编辑器扩展(二)——定义自己的窗口 Unity3D编辑器扩展(三)—— ...

  2. Unity3D编辑器扩展(五)——常用特性(Attribute)以及Selection类

    前面写了四篇关于编辑器的: Unity3D编辑器扩展(一)——定义自己的菜单按钮 Unity3D编辑器扩展(二)——定义自己的窗口 Unity3D编辑器扩展(三)——使用GUI绘制窗口 Unity3D ...

  3. Unity3D编辑器扩展(四)——扩展自己的组件

    前面已经写了三篇: Unity3D编辑器扩展(一)——定义自己的菜单按钮 Unity3D编辑器扩展(二)——定义自己的窗口 Unity3D编辑器扩展(三)——使用GUI绘制窗口 今天写第四篇,扩展自己 ...

  4. Unity3D之UGUI学习笔记(一):UGUI介绍以及Canvas

    UGUI是Unity3D4.6官方提供的UI系统,支持2D和3D UI的开发. Unity3D UI史 OnGUI 在Unity4.6之前,官方提供的是OnGUI函数来开发UI界面,当然问题也比较多, ...

  5. Unity3D之UGUI学习笔记(三):EventSystem

    在UGUI中,EventSystem实现了所有关于交互方面的功能,和NGUI不一样的地方是,我们终于可以摆脱添加Box Collider了! 下面我们来学习一下. 对于按钮来说,直接有onClick的 ...

  6. unity3D编辑器扩展

    编辑器扩展只是在编辑项目中运行,发布出来是不会运行的. 固定创建一个文件夹Editor:所有的资源或者代码都不会被打包进去. 01.使用MenuItem添加菜单栏按钮 脚本不需要作为组件存在,可以不用 ...

  7. 【第五课】VIM编辑器(学习笔记)

    4月10日学习笔记打卡

  8. Unity3D编辑器扩展(一)——定义自己的菜单按钮

    Unity3D 引擎的编辑器拥有很强的扩展性,用的好可以帮我们省很多事情.在这里记录下如何去扩展 Unity3D 的编辑器,定制属于我们自己的开发环境. 本篇主要讲解在 Unity3D 引擎的各个窗口 ...

  9. Unity3D之ScriptableObject学习笔记

    不同与C#提供的Serializable序列化功能,ScriptableObject是Unity3D提供的一个数据存储类,我们接下来学习一下这个类的功能. 官方文档 http://docs.unity ...

随机推荐

  1. xaml mvvm(1)之结构

    在微软winstore.wp和silverlight中xaml是用来构建UI视图的标记语言,全名Extensible Application Markup Language.在结构上类似于html,但 ...

  2. Foxman, 基于微核架构的 Mock 解决方案

    本文来自 网易云社区 . Foxman ⇗ 是一个使用 Node.js 开发的命令行工具,定位是一个可扩展的 Mock Server,帮助前端开发者轻松.独立.高效地进行前端开发和完成后续的联调工作. ...

  3. 【Selenium专题】 FAQ_对象识别_Compound class names are not supported

    测试代码 public void login(){ WebDriver driver = new ChromeDriver(); driver.get("http://IP:Port/cli ...

  4. CookieJar和HTTPCookieProcessor

    CookieJar和HTTPCookieProcessor 我们在使用爬虫的时候,经常会用到cookie进行模拟登陆和访问.在使用urllib库做爬虫,我们需要借助http.cookiejar库中的C ...

  5. bzoj1009GT考试

    题目链接 没啥好说的,矩阵优化+$kmp$字符串匹配 上代码: /************************************************************** Prob ...

  6. API自动化测试 Soap UI工具介绍

    一.   建立测试用例 (一)   基本概念 soapUI 中工程的层次结构 项目名称:位于最上层 (BookStoreTest),项目可以包含多个服务的定义. REST 服务定义:服务其实是对多个 ...

  7. 补丁patch 漏洞 bug或glitch

    补丁patch漏洞 bug或glitch    

  8. springboot 搭建 简单 web项目 【springboot + freemark模板 + yml 配置文件 + 热修复 + 测试用例】附源码

    项目 地址:  https://gitee.com/sanmubird/springboot-simpleweb.git 项目介绍: 本项目主要有一下内容: 1: springboot yml 配置 ...

  9. nginx高性能WEB服务器系列之九--nginx运维故障日常解决方案

    nginx系列友情链接:nginx高性能WEB服务器系列之一简介及安装https://www.cnblogs.com/maxtgood/p/9597596.htmlnginx高性能WEB服务器系列之二 ...

  10. 四大算法解决最短路径问题(Dijkstra+Bellman-ford+SPFA+Floyd)

    什么是最短路径问题? 简单来讲,就是用于计算一个节点到其他所有节点的最短路径. 单源最短路算法:已知起点,求到达其他点的最短路径. 常用算法:Dijkstra算法.Bellman-ford算法.SPF ...