#if UNITY_EDITOR using DCFApixels.DragonECS.Unity.Internal; using System; using System.Collections.Generic; using System.Reflection; using System.Runtime.InteropServices; using System.Text; using UnityEditor; using UnityEngine; namespace DCFApixels.DragonECS.Unity.Editors { internal static class EcsUnityEditorUtility { public static string TransformFieldName(string name) { if (name.Length <= 0) { return name; } StringBuilder b = new StringBuilder(); bool nextWorld = true; bool prewIsUpper = false; for (int i = 0; i < name.Length; i++) { char c = name[i]; if (char.IsLetter(c) == false) { nextWorld = true; prewIsUpper = false; continue; } bool isUpper = char.IsUpper(c); if (isUpper) { if (nextWorld == false && prewIsUpper == false) { b.Append(' '); nextWorld = true; } } if (nextWorld) { b.Append(char.ToUpper(c)); } else { b.Append(c); } nextWorld = false; prewIsUpper = isUpper; } return b.ToString(); } } public static class EcsGUI { private static Color _grayColor = new Color32(100, 100, 100, 100); private static Color _greenColor = new Color32(75, 255, 0, 100); private static Color _redColor = new Color32(255, 0, 75, 100); private static GUIStyle _grayStyle; private static GUIStyle _greenStyle; private static GUIStyle _redStyle; private static GUILayoutOption[] _defaultParams; private static bool _isInit = false; private static void Init() { if (_isInit) { return; } _defaultParams = new GUILayoutOption[] { GUILayout.ExpandWidth(true) }; _grayStyle = EcsEditor.GetStyle(_grayColor); _greenStyle = EcsEditor.GetStyle(_greenColor); _redStyle = EcsEditor.GetStyle(_redColor); _isInit = true; } private const string CONNECTED = "Connected"; private const string NOT_CONNECTED = "Not connected"; private const string UNDETERMINED_CONNECTED = "---"; public static void DrawConnectStatus(Rect position, bool status) { Init(); if (status) { GUI.Box(position, CONNECTED, _greenStyle); } else { GUI.Box(position, NOT_CONNECTED, _redStyle); } } public static void DrawUndeterminedConnectStatus(Rect position) { Init(); GUI.Box(position, UNDETERMINED_CONNECTED, _grayStyle); } public static class Layout { public static void DrawConnectStatus(bool status, params GUILayoutOption[] options) { Init(); if (options == null || options.Length <= 0) { options = _defaultParams; } GUILayout.Box("", options); Rect lastRect = GUILayoutUtility.GetLastRect(); Color color = status ? _greenColor : _redColor; string text = status ? CONNECTED : NOT_CONNECTED; color.a = 0.6f; EditorGUI.DrawRect(lastRect, color); GUI.Box(lastRect, text); } public static void DrawUndeterminedConnectStatus(params GUILayoutOption[] options) { Init(); if (options == null || options.Length <= 0) { options = _defaultParams; } GUILayout.Box(UNDETERMINED_CONNECTED, _grayStyle, options); } public static void DrawComponents(entlong entity) { if (entity.TryUnpack(out int entityID, out EcsWorld world)) { DrawComponents(entityID, world); } } public static void DrawComponents(int entityID, EcsWorld world) { var componentTypeIDs = world.GetComponentTypeIDs(entityID); foreach (var componentTypeID in componentTypeIDs) { var pool = world.GetPool(componentTypeID); { DrawComponent(entityID, world, pool); } } } private static readonly BindingFlags fieldFlags = BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic; private static void DrawComponent(int entityID, EcsWorld world, IEcsPool pool) { object data = pool.GetRaw(entityID); var meta = data.GetMeta(); Color panelColor = meta.Color.ToUnityColor(); GUILayout.BeginVertical(EcsEditor.GetStyle(panelColor, 0.22f)); EditorGUI.BeginChangeCheck(); Type componentType = pool.ComponentType; ExpandMatrix expandMatrix = ExpandMatrix.Take(componentType); bool changed = DrawData(componentType, new GUIContent(meta.Name), expandMatrix, data, out object resultData); if (changed) { pool.SetRaw(entityID, resultData); } GUILayout.EndVertical(); Rect lineRect = GUILayoutUtility.GetLastRect(); lineRect.y = lineRect.yMax; lineRect.height = 3f; Color rectColor = panelColor; rectColor.a = 0.34f; EditorGUI.DrawRect(lineRect, rectColor); GUILayout.Space(2f); } private static bool DrawData(Type fieldType, GUIContent label, ExpandMatrix expandMatrix, object data, out object outData) { Type type = data.GetType(); var uobj = data as UnityEngine.Object; ref bool isExpanded = ref expandMatrix.Down(); if ((uobj == false && type.IsGenericType) || (uobj == false && !type.IsSerializable)) { bool result = false; isExpanded = EditorGUILayout.Foldout(isExpanded, label); if (isExpanded) { EditorGUI.indentLevel++; foreach (var field in type.GetFields(fieldFlags)) { GUIContent subLabel = new GUIContent(EcsUnityEditorUtility.TransformFieldName(field.Name)); if (DrawData(field.FieldType, subLabel, expandMatrix, field.GetValue(data), out object fieldData)) { field.SetValue(data, fieldData); result = true; } } EditorGUI.indentLevel--; } expandMatrix.Up(); outData = data; return result; } else { EditorGUI.BeginChangeCheck(); WrapperBase w = uobj == null ? RefEditorWrapper.Take(data) : UnityObjEditorWrapper.Take(uobj); w.IsExpanded = isExpanded; EditorGUILayout.PropertyField(w.Property, label, true); isExpanded = w.IsExpanded; w.Release(); expandMatrix.Up(); if (EditorGUI.EndChangeCheck()) { w.SO.ApplyModifiedProperties(); outData = w.Data; return true; } outData = data; return false; } } } } [InitializeOnLoad] public static class EcsEditor { static EcsEditor() { colorBoxeStyles = new SparseArray(); } private static SparseArray colorBoxeStyles = new SparseArray(); public static GUIStyle GetStyle(Color color, float alphaMultiplier) { color.a *= alphaMultiplier; return GetStyle(color); } public static GUIStyle GetStyle(Color32 color32) { int colorCode = new Color32Union(color32).colorCode; if (colorBoxeStyles.TryGetValue(colorCode, out GUIStyle style)) { if (style == null || style.normal.background == null) { style = CreateStyle(color32, colorCode); colorBoxeStyles[colorCode] = style; } return style; } style = CreateStyle(color32, colorCode); colorBoxeStyles.Add(colorCode, style); return style; } private static GUIStyle CreateStyle(Color32 color32, int colorCode) { GUIStyle result = new GUIStyle(GUI.skin.box); Color componentColor = color32; Texture2D texture2D = CreateTexture(2, 2, componentColor); result.hover.background = texture2D; result.focused.background = texture2D; result.active.background = texture2D; result.normal.background = texture2D; return result; } private static Texture2D CreateTexture(int width, int height, Color color) { var pixels = new Color[width * height]; for (var i = 0; i < pixels.Length; ++i) pixels[i] = color; var result = new Texture2D(width, height); result.SetPixels(pixels); result.Apply(); return result; } #region Utils [StructLayout(LayoutKind.Explicit, Pack = 1, Size = 4)] private readonly ref struct Color32Union { [FieldOffset(0)] public readonly int colorCode; [FieldOffset(0)] public readonly byte r; [FieldOffset(1)] public readonly byte g; [FieldOffset(2)] public readonly byte b; [FieldOffset(3)] public readonly byte a; public Color32Union(byte r, byte g, byte b, byte a) : this() { this.r = r; this.g = g; this.b = b; this.a = a; } public Color32Union(Color32 color) : this() { r = color.r; g = color.g; b = color.b; a = color.a; } } #endregion } public static class ReflectionExtensions { public static bool TryGetAttribute(this MemberInfo self, out T attrbiute) where T : Attribute { attrbiute = self.GetCustomAttribute(); return attrbiute != null; } public static bool HasAttribute(this MemberInfo self) where T : Attribute { return self.GetCustomAttribute() != null; } } } #endif