Unity 编辑器开发实战【Custom Editor】- FSM Editor

举报
CoderZ1010 发表于 2022/09/25 04:43:27 2022/09/25
【摘要】         本文介绍如何为FSM有限状态机模块实现一个自定义编辑器面板,FSM的详细代码在上一篇文章中有介绍,链接地址: 在Unity中构建FSM有限状态机 下面是最终效果:     &n...

        本文介绍如何为FSM有限状态机模块实现一个自定义编辑器面板,FSM的详细代码在上一篇文章中有介绍,链接地址: 在Unity中构建FSM有限状态机 下面是最终效果:

        首先,自定义一个编辑器面板,需要用到Attribute:CustomEditor,参数传入目标类的类型,代码如下:


  
  1. using UnityEngine;
  2. using UnityEditor;
  3. namespace SK.Framework
  4. {
  5. [CustomEditor(typeof(FSMMaster))]
  6. public class FSMEditor : Editor
  7. {
  8. }
  9. }

        自定义编辑器类继承Editor类后,重写OnInspectorGUI函数来自定义Inspector面板,例如添加一个Label文本:


  
  1. using UnityEngine;
  2. using UnityEditor;
  3. namespace SK.Framework
  4. {
  5. [CustomEditor(typeof(FSMMaster))]
  6. public class FSMEditor : Editor
  7. {
  8. public override void OnInspectorGUI()
  9. {
  10. GUILayout.Label("有限状态机");
  11. }
  12. }
  13. }

        绘制该面板我们需要FSM Master中的状态机列表的信息,是一个私有的StateMachine类型的列表,因此需要通过反射去获取:


  
  1. using System.Reflection;
  2. using System.Collections.Generic;
  3. using UnityEngine;
  4. using UnityEditor;
  5. namespace SK.Framework
  6. {
  7. [CustomEditor(typeof(FSMMaster))]
  8. public class FSMEditor : Editor
  9. {
  10. private List<StateMachine> machines;
  11. public override void OnInspectorGUI()
  12. {
  13. //程序未在运行状态则退出
  14. if (!Application.isPlaying) return;
  15. if (machines == null)
  16. {
  17. //通过反射获取状态机列表
  18. machines = typeof(FSMMaster).GetField("machines", BindingFlags.Instance | BindingFlags.NonPublic)
  19. .GetValue(FSMMaster.Instance) as List<StateMachine>;
  20. }
  21. }
  22. }
  23. }

         有了状态机的信息后,通过EditorGUILayout类中的Popup去列举所有的状态机,其中需要传入一个string类型数组,即列举的内容,我们声明一个string类型数组来存储所有状态机的名称,使用一个int类型字段来表示当前选中的状态机的索引:


  
  1. using System.Linq;
  2. using System.Reflection;
  3. using System.Collections.Generic;
  4. using UnityEngine;
  5. using UnityEditor;
  6. namespace SK.Framework
  7. {
  8. [CustomEditor(typeof(FSMMaster))]
  9. public class FSMEditor : Editor
  10. {
  11. private List<StateMachine> machines;
  12. private int currentMachineIndex;
  13. private string[] machinesName;
  14. public override void OnInspectorGUI()
  15. {
  16. //程序未在运行状态则退出
  17. if (!Application.isPlaying) return;
  18. if (machines == null)
  19. {
  20. //通过反射获取状态机列表
  21. machines = typeof(FSMMaster).GetField("machines", BindingFlags.Instance | BindingFlags.NonPublic)
  22. .GetValue(FSMMaster.Instance) as List<StateMachine>;
  23. }
  24. //当状态机名称数组为空(初始化) 或数量与状态机数量不等时(状态机列表发生变化)
  25. if (machinesName == null || machines.Count != machinesName.Length)
  26. {
  27. //重置当前状态机索引数值
  28. currentMachineIndex = 0;
  29. //重新获取状态机名称数组
  30. machinesName = machines.Select(m => m.Name).ToArray();
  31. }
  32. if (machines.Count > 0)
  33. {
  34. currentMachineIndex = EditorGUILayout.Popup("状态机:", currentMachineIndex, machinesName);
  35. }
  36. }
  37. }
  38. }

        接下来获取状态机中的所有状态信息, 状态使用一个IState类型的列表存储,修饰符为protected,因此也通过反射去获取:


  
  1. using System.Linq;
  2. using System.Reflection;
  3. using System.Collections.Generic;
  4. using UnityEngine;
  5. using UnityEditor;
  6. namespace SK.Framework
  7. {
  8. [CustomEditor(typeof(FSMMaster))]
  9. public class FSMEditor : Editor
  10. {
  11. private List<StateMachine> machines;
  12. private FieldInfo statesFieldInfo;
  13. private int currentMachineIndex;
  14. private string[] machinesName;
  15. public override void OnInspectorGUI()
  16. {
  17. //程序未在运行状态则退出
  18. if (!Application.isPlaying) return;
  19. if (machines == null)
  20. {
  21. //通过反射获取状态机列表
  22. machines = typeof(FSMMaster).GetField("machines", BindingFlags.Instance | BindingFlags.NonPublic)
  23. .GetValue(FSMMaster.Instance) as List<StateMachine>;
  24. //获取状态列表字段
  25. statesFieldInfo = typeof(StateMachine).GetField("states", BindingFlags.Instance | BindingFlags.NonPublic);
  26. }
  27. //当状态机名称数组为空(初始化) 或数量与状态机数量不等时(状态机列表发生变化)
  28. if (machinesName == null || machines.Count != machinesName.Length)
  29. {
  30. //重置当前状态机索引数值
  31. currentMachineIndex = 0;
  32. //重新获取状态机名称数组
  33. machinesName = machines.Select(m => m.Name).ToArray();
  34. }
  35. if (machines.Count > 0)
  36. {
  37. currentMachineIndex = EditorGUILayout.Popup("状态机:", currentMachineIndex, machinesName);
  38. var currentMachine = machines[currentMachineIndex];
  39. //获取当前状态机的状态列表
  40. var states = statesFieldInfo.GetValue(currentMachine) as List<IState>;
  41. }
  42. }
  43. }
  44. }

        有了状态的列表信息后,for循环遍历列表,绘制每一个状态的名称,使用不同的GUIStyle来区分该状态是否为状态机的当前状态,如果不是,则提供一个切换到该状态的Button按钮:


  
  1. using System.Linq;
  2. using System.Reflection;
  3. using System.Collections.Generic;
  4. using UnityEngine;
  5. using UnityEditor;
  6. namespace SK.Framework
  7. {
  8. [CustomEditor(typeof(FSMMaster))]
  9. public class FSMEditor : Editor
  10. {
  11. private List<StateMachine> machines;
  12. private FieldInfo statesFieldInfo;
  13. private int currentMachineIndex;
  14. private string[] machinesName;
  15. public override void OnInspectorGUI()
  16. {
  17. //程序未在运行状态则退出
  18. if (!Application.isPlaying) return;
  19. if (machines == null)
  20. {
  21. //通过反射获取状态机列表
  22. machines = typeof(FSMMaster).GetField("machines", BindingFlags.Instance | BindingFlags.NonPublic)
  23. .GetValue(FSMMaster.Instance) as List<StateMachine>;
  24. //获取状态列表字段
  25. statesFieldInfo = typeof(StateMachine).GetField("states", BindingFlags.Instance | BindingFlags.NonPublic);
  26. }
  27. //当状态机名称数组为空(初始化) 或数量与状态机数量不等时(状态机列表发生变化)
  28. if (machinesName == null || machines.Count != machinesName.Length)
  29. {
  30. //重置当前状态机索引数值
  31. currentMachineIndex = 0;
  32. //重新获取状态机名称数组
  33. machinesName = machines.Select(m => m.Name).ToArray();
  34. }
  35. if (machines.Count > 0)
  36. {
  37. currentMachineIndex = EditorGUILayout.Popup("状态机:", currentMachineIndex, machinesName);
  38. var currentMachine = machines[currentMachineIndex];
  39. //获取当前状态机的状态列表
  40. var states = statesFieldInfo.GetValue(currentMachine) as List<IState>;
  41. GUILayout.BeginVertical("Box");
  42. for (int i = 0; i < states.Count; i++)
  43. {
  44. var state = states[i];
  45. //如果状态为当前状态 使用SelectionRect Style 否则使用IN Title Style进行区分
  46. GUILayout.BeginHorizontal(currentMachine.CurrentState == state ? "SelectionRect" : "IN Title");
  47. GUILayout.Label(state.Name);
  48. //如果状态不是当前状态 提供切换到该状态的Button按钮
  49. if(currentMachine.CurrentState != state)
  50. {
  51. if (GUILayout.Button("Switch", GUILayout.Width(50f)))
  52. {
  53. currentMachine.Switch(state);
  54. }
  55. }
  56. GUILayout.EndHorizontal();
  57. }
  58. GUILayout.EndVertical();
  59. }
  60. }
  61. }
  62. }

        除此之外,我们还希望在状态机下面添加一排菜单,绘制三个按钮,分别实现状态机中的切换到下一状态、切换到上一状态、切换到空状态的功能,通过GUILayout类中的BeginHorizontal和EndHorizontal将这三个按钮绘制到一排:


  
  1. private class GUIContents
  2. {
  3. public static GUIContent switch2Next = new GUIContent("Next", "切换到下一状态");
  4. public static GUIContent switch2Last = new GUIContent("Last", "切换到上一状态");
  5. public static GUIContent switch2Null = new GUIContent("Null", "切换到空状态 (退出当前状态)");
  6. }

  
  1. GUILayout.BeginHorizontal();
  2. //提供切换到上一状态的Button按钮
  3. if (GUILayout.Button(GUIContents.switch2Last, "ButtonLeft"))
  4. {
  5. currentMachine.Switch2Last();
  6. }
  7. //提供切换到下一状态的Button按钮
  8. if (GUILayout.Button(GUIContents.switch2Next, "ButtonMid"))
  9. {
  10. currentMachine.Switch2Next();
  11. }
  12. //提供切换到空状态的Button按钮
  13. if (GUILayout.Button(GUIContents.switch2Null, "ButtonRight"))
  14. {
  15. currentMachine.Switch2Null();
  16. }
  17. GUILayout.EndHorizontal();

最终完整代码:


  
  1. using System.Linq;
  2. using System.Reflection;
  3. using System.Collections.Generic;
  4. using UnityEngine;
  5. using UnityEditor;
  6. namespace SK.Framework
  7. {
  8. [CustomEditor(typeof(FSMMaster))]
  9. public class FSMEditor : Editor
  10. {
  11. private class GUIContents
  12. {
  13. public static GUIContent switch2Next = new GUIContent("Next", "切换到下一状态");
  14. public static GUIContent switch2Last = new GUIContent("Last", "切换到上一状态");
  15. public static GUIContent switch2Null = new GUIContent("Null", "切换到空状态 (退出当前状态)");
  16. }
  17. private List<StateMachine> machines;
  18. private FieldInfo statesFieldInfo;
  19. private int currentMachineIndex;
  20. private string[] machinesName;
  21. public override void OnInspectorGUI()
  22. {
  23. //程序未在运行状态则退出
  24. if (!Application.isPlaying) return;
  25. if (machines == null)
  26. {
  27. //通过反射获取状态机列表
  28. machines = typeof(FSMMaster).GetField("machines", BindingFlags.Instance | BindingFlags.NonPublic)
  29. .GetValue(FSMMaster.Instance) as List<StateMachine>;
  30. //获取状态列表字段
  31. statesFieldInfo = typeof(StateMachine).GetField("states", BindingFlags.Instance | BindingFlags.NonPublic);
  32. }
  33. //当状态机名称数组为空(初始化) 或数量与状态机数量不等时(状态机列表发生变化)
  34. if (machinesName == null || machines.Count != machinesName.Length)
  35. {
  36. //重置当前状态机索引数值
  37. currentMachineIndex = 0;
  38. //重新获取状态机名称数组
  39. machinesName = machines.Select(m => m.Name).ToArray();
  40. }
  41. if (machines.Count > 0)
  42. {
  43. currentMachineIndex = EditorGUILayout.Popup("状态机:", currentMachineIndex, machinesName);
  44. var currentMachine = machines[currentMachineIndex];
  45. //获取当前状态机的状态列表
  46. var states = statesFieldInfo.GetValue(currentMachine) as List<IState>;
  47. GUILayout.BeginHorizontal();
  48. //提供切换到上一状态的Button按钮
  49. if (GUILayout.Button(GUIContents.switch2Last, "ButtonLeft"))
  50. {
  51. currentMachine.Switch2Last();
  52. }
  53. //提供切换到下一状态的Button按钮
  54. if (GUILayout.Button(GUIContents.switch2Next, "ButtonMid"))
  55. {
  56. currentMachine.Switch2Next();
  57. }
  58. //提供切换到空状态的Button按钮
  59. if (GUILayout.Button(GUIContents.switch2Null, "ButtonRight"))
  60. {
  61. currentMachine.Switch2Null();
  62. }
  63. GUILayout.EndHorizontal();
  64. GUILayout.BeginVertical("Box");
  65. for (int i = 0; i < states.Count; i++)
  66. {
  67. var state = states[i];
  68. //如果状态为当前状态 使用SelectionRect Style 否则使用IN Title Style进行区分
  69. GUILayout.BeginHorizontal(currentMachine.CurrentState == state ? "SelectionRect" : "IN Title");
  70. GUILayout.Label(state.Name);
  71. //如果状态不是当前状态 提供切换到该状态的Button按钮
  72. if(currentMachine.CurrentState != state)
  73. {
  74. if (GUILayout.Button("Switch", GUILayout.Width(50f)))
  75. {
  76. currentMachine.Switch(state);
  77. }
  78. }
  79. GUILayout.EndHorizontal();
  80. }
  81. GUILayout.EndVertical();
  82. }
  83. }
  84. }
  85. }

文章来源: coderz.blog.csdn.net,作者:CoderZ1010,版权归原作者所有,如需转载,请联系作者。

原文链接:coderz.blog.csdn.net/article/details/122488190

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。