第三章自定义检视面板_创建自定义编辑器类_编扩展默认组件的显示面板(本章进度6/9)

3.1.5 扩展默认组件的显示面板

为Unity 的 RectTransform 组件添加了一个​​一键锚点自动设置工具​​,在保留 Unity 内置编辑器所有功能的基础上,增加了实用功能。

csharp 复制代码
// 引入必要的命名空间
using System;
using System.Linq;
using System.Reflection;
using UnityEditor;
using UnityEngine;

/// <summary>
/// 声明自定义编辑器,作用于RectTransform组件
/// 一键自动计算并设置锚点
//保持 UI 元素在父容器中的相对位置
//自动重置偏移量为零
/// </summary>
[CustomEditor(typeof(RectTransform))]
public class RectTransformEditorExample : Editor
{
    private RectTransform rt;         // 当前编辑的RectTransform组件
    private Editor instance;          // 内置RectTransform编辑器的实例
    private MethodInfo onSceneGUI;    // 内置编辑器的OnSceneGUI方法引用
    private static readonly object[] emptyArray = new object[0];  // 空参数数组用于反射调用

    // 初始化方法
    private void OnEnable()
    {
        rt = target as RectTransform;
        if (rt == null)
            return;

        // 反射获取Unity内置的RectTransformEditor类型
        var editorType = Assembly.GetAssembly(typeof(Editor))  // 获取Editor所在程序集
                      .GetTypes()                             // 获取程序集中所有类型
                      .FirstOrDefault(                        // 查找符合条件的第一个类型
                          m => m.Name == "RectTransformEditor" // 根据名称匹配内置编辑器
                      );

        // 创建内置编辑器实例
        instance = CreateEditor(targets, editorType);

        // 反射获取内置编辑器的私有OnSceneGUI方法
        onSceneGUI = editorType.GetMethod(
            "OnSceneGUI",
            BindingFlags.Instance | BindingFlags.NonPublic); // 
    }

    // 场景视图绘制
    private void OnSceneGUI()
    {
        // 调用内置编辑器的场景视图绘制方法
        onSceneGUI?.Invoke(instance, emptyArray); 
    }

    // 清理资源
    private void OnDisable()
    {
        if (instance != null)
        {
            DestroyImmediate(instance); // 销毁内置编辑器实例
        }
    }

    // 绘制Inspector面板
    public override void OnInspectorGUI()
    {
        // 先绘制默认的Inspector内容
        instance.OnInspectorGUI();
        // 添加自定义的锚点工具
        OnAnchorSetHelperGUI();
    }

    // 自定义的锚点工具GUI
    private void OnAnchorSetHelperGUI()
    {
        EditorGUILayout.Space();

        // 使用醒目颜色标题
        Color color = GUI.color;
        GUI.color = Color.cyan;
        GUILayout.Label("锚点工具", EditorStyles.boldLabel);
        GUI.color = color;

        // Auto Anchor按钮
        if (GUILayout.Button("Auto Anchor"))
        {
            // 注册撤销操作
            Undo.RecordObject(rt, "Auto Anchor");

            // 获取父物体的RectTransform
            RectTransform prt = rt.parent as RectTransform;
            if (prt == null)
                return;

            // 核心算法:计算新的锚点位置
            Vector2 anchorMin = new(
                rt.anchorMin.x + rt.offsetMin.x / prt.rect.width,
                rt.anchorMin.y + rt.offsetMin.y / prt.rect.height
            );

            Vector2 anchorMax = new(
                rt.anchorMax.x + rt.offsetMax.x / prt.rect.width,
                rt.anchorMax.y + rt.offsetMax.y / prt.rect.height
            );

            // 应用新锚点并重置偏移量
            rt.anchorMin = anchorMin;
            rt.anchorMax = anchorMax;
            rt.offsetMin = Vector2.zero;
            rt.offsetMax = Vector2.zero;
        }
    }
}

效果是这样的,当我们移动UI控件时,锚点是不会跟着移动的,需要我们用预设好的锚点位置,或者手动调整,一下比较麻烦一些,如下图。

点击Auto Anchor ,会自动设置锚点,如下图

相关推荐
布兰妮甜2 小时前
Visual Studio Code 跨平台快捷键指南:Windows 与 macOS 全面对比
windows·vscode·macos·编辑器·快捷键
凯子坚持 c1 天前
虚拟机一站式部署Claude Code &可视化UI界面
ui·编辑器·vim
fananchong21 天前
制作 VSCode 插件
ide·vscode·编辑器·ai辅助编程·vscode 插件
研究是为了理解2 天前
Linux Shell:Nano 编辑器备忘
linux·运维·编辑器
下水道的臭老鼠2 天前
vscode+latex本地英文期刊环境配置
ide·vscode·编辑器
NKelly3 天前
原生Vim操作大全
linux·编辑器·vim
惟余惟余3 天前
VBOX虚拟机使用指导
运维·编辑器
guidovans3 天前
node.js 零基础入门
node.js·编辑器·vim
斯幽柏雷科技4 天前
[已解决]VSCode右键菜单消失恢复
ide·vscode·编辑器
时空自由民.5 天前
rk3588s vscode索引失败的问题
ide·vscode·编辑器