【Unity3d编辑器从入门到精通】深入简出聊Undo

简单演示Undo

Undo的演示

Undo的原理 LIFO

原理
变色

简单实现

创建物体Undo.RegisterCreatedObjectUndo

    [MenuItem("Example/Create Cube")]
    static void CreateCube()
    {
        var cube = GameObject.CreatePrimitive(PrimitiveType.Cube);
        Undo.RegisterCreatedObjectUndo(cube, "Create Cube");
    }
执行效果

属性变化RecordObject/RegisterCompleteObjectUndo

    [MenuItem("Example/Random Rotate")]
    static void RandomRotate ()
    {
        var transform = Selection.activeTransform;

        if (transform) {
            Undo.RecordObject (transform, "Rotate " + transform.name);
            transform.rotation = Random.rotation;
        }
    }

Unity又是怎么处理的呢?PropertyDiffUndoRecorder

使用Profiler检查PropertyDiffUndoRecorder

执行顺序

  • RecordObject
  • 处理属性
  • Flush
  • 输出差异
差异示意图

差异的代码查看

    [MenuItem("Example/Random Rotate")]
    static void RandomRotate()
    {
        var transform = Selection.activeTransform;
        if (transform)
        {
            //Undo.RecordObject(transform,"Rotate " + transform.name);
            //Undo.RegisterCompleteObjectUndo(transform, "Rotate " + transform.name);
            Undo.willFlushUndoRecord += () => Debug.Log("flush");
            Undo.postprocessModifications += (modifications) =>
            {
                Debug.Log("modifications");
                return modifications;
            };
            Undo.RecordObject(transform, "Rotate" + transform.name);
            Debug.Log("record");

            transform.rotation = Random.rotation;
            Debug.Log("changed");
        }
    }
查看Debug
Debug

Redo的实现

正常执行
Undo
Redo

Undo处理的对象

Undo定位的是一个可序列化的对象,继承自UnityEngine.Object

Undo支持的对象

GameObject
Component (MonoBehaviour)
ScriptableObject

System.Serializable序列化后可以被Undo

// 序列化
[System.Serializable]
public class PlayerInfo
{
    public string name;
    public int hp;
}

public class PlayerRedo : MonoBehaviour
{
    [SerializeField]
    public PlayerInfo info;
}
// 调用
public class ExampleRedo {
    [MenuItem("Example/Change PlayerInfo")]
    static void ChangePlayerInfo()
    {
        var player = Selection.activeGameObject.GetComponent<PlayerRedo>();

        if (player)
        {
            Undo.RecordObject(player, "Change PlayerInfo");
            player.info = new PlayerInfo
            {
                name = "New PlayerName",
                hp = Random.Range(0, 10)
            };
        }
    }
}

Undo处理的类型

  • 对象属性
  • 对象本身的操作

属性撤销[可以实现绝大部分功能]

[MenuItem("Undo/RecordObject测试")]
    static void RecordObject()
    {
        Transform transform = Selection.activeTransform;

        Undo.RecordObject(transform, "position 指定变更为 Vector3(0,0,0)");
        transform.position = new Vector3(0,0,0);
    }
自定义的名字最好是英文,对中文字符支持不好

一些自带的Undo

    [MenuItem("Undo/AddComponent")]
    static void AddComponent()
    {
        GameObject go = Selection.activeGameObject;
        Rigidbody rigidbody = Undo.AddComponent<Rigidbody>(go);
    }

    [MenuItem("Undo/RegisterCreatedObjectUndo")]
    static void RegisterCreatedObjectUndo()
    {
        GameObject go = new GameObject();
        Undo.RegisterCreatedObjectUndo(go, "GameObject 生成");
        // 组的概念
        Undo.IncrementCurrentGroup();

        ScriptableRedo scriptableRedo = ScriptableObject.CreateInstance<ScriptableRedo>();
        Undo.RegisterCreatedObjectUndo(scriptableRedo, "ScriptableObject 生成");

        //EditorApplication.update += () => Debug.Log(scriptableRedo);
    }

    [MenuItem("Undo/DestoryObjectImmediate")]
    static void DestoryObjectImmediate()
    {
        GameObject go = Selection.activeGameObject;
        Undo.DestroyObjectImmediate(go);
    }

    [MenuItem("Undo/SetTransformParent")]
    static void SetTransformParent()
    {
        Transform root = GameObject.Find("Main Camera").transform;
        Transform transform = Selection.activeTransform;
        transform.SetParent(root);
        //Undo.SetTransformParent(transform, root, "摄像机子物体");
    }

Revert 表示不想要恢复

Revert

Undo.RevertAllInCurrentGroup Revert当前组

    [MenuItem("Undo/RevertAllInCurrentGroup")]
    static void RevertAllInCurrentGroup()
    {
        GameObject ticket = new GameObject("Ticket");
        Undo.RegisterCreatedObjectUndo(ticket,"Ticket");

        int number = ticket.GetInstanceID();

        int winningNumber = 1234;

        if (number != winningNumber)
        {
            Undo.RevertAllInCurrentGroup();
        }
    }

Undo.RevertAllDownToGroup Revert到指定的组索引

//RevertAllDownToGroup
public class ExampleRevertUndoWindow : EditorWindow
{
    [MenuItem("Undo/Window/RevertUndoWindow")]
    static void Open()
    {
        GetWindow<ExampleRevertUndoWindow>();
    }

    private GameObject go;

    private int group1 = 0;
    private int group2 = 0;
    private int group3 = 0;

    void OnEnable()
    {
        go = GameObject.Find("New Game Object");
    }

    void OnGUI()
    {
        if (Event.current.type == EventType.MouseDown)
        {
            group1 = Undo.GetCurrentGroup();

            Undo.AddComponent<Rigidbody>(go);

            Undo.IncrementCurrentGroup();

            group2 = Undo.GetCurrentGroup();

            Undo.AddComponent<BoxCollider>(go);

            Undo.IncrementCurrentGroup();

            group3 = Undo.GetCurrentGroup();

            Undo.AddComponent<ConstantForce>(go);
        }

        if (Event.current.type == EventType.MouseUp)
        {
            Undo.RevertAllDownToGroup(group2);

            EditorGUIUtility.ExitGUI();
        }
    }
}

组的概念

一般写到一段处理代码里面的操作,被划为一组

Undo.IncrementCurrentGroup 单独执行每个撤消

    [MenuItem("Undo/RegisterCreatedObjectUndo")]
    static void RegisterCreatedObjectUndo()
    {
        GameObject go = new GameObject();
        Undo.RegisterCreatedObjectUndo(go, "GameObject 生成");
        // 组的概念
        Undo.IncrementCurrentGroup();

        ScriptableRedo scriptableRedo = ScriptableObject.CreateInstance<ScriptableRedo>();
        Undo.RegisterCreatedObjectUndo(scriptableRedo, "ScriptableObject 生成");

        //EditorApplication.update += () => Debug.Log(scriptableRedo);
    }

CollapseUndoOperations 组合多个Undo

//CollapseUndo
public class ExampleCollapseUndo : EditorWindow
{
    private int groupID = 0;
    [MenuItem("Undo/Window/CollapseUndoWindow")]
    static void Open()
    {
        GetWindow<ExampleCollapseUndo>();
    }

    void OnEnbable()
    {
        groupID = Undo.GetCurrentGroup();
    }

    void OnDisable()
    {
        Undo.CollapseUndoOperations(groupID);
    }

    void OnGUI()
    {
        if (GUILayout.Button("cube"))
        {
            var cube = GameObject.CreatePrimitive(PrimitiveType.Cube);
            Undo.RegisterCreatedObjectUndo(cube, "Create cube");
        }

        if (GUILayout.Button("plane"))
        {
            var plane = GameObject.CreatePrimitive(PrimitiveType.Plane);
            Undo.RegisterCreatedObjectUndo(plane, "Create plane");
        }

        if (GUILayout.Button("Cylinder"))
        {
            var cylinder = GameObject.CreatePrimitive(PrimitiveType.Cylinder);
            Undo.RegisterCreatedObjectUndo(cylinder, "Create cylinder");
        }
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,864评论 6 494
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,175评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,401评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,170评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,276评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,364评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,401评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,179评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,604评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,902评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,070评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,751评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,380评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,077评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,312评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,924评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,957评论 2 351

推荐阅读更多精彩内容

  • 国家电网公司企业标准(Q/GDW)- 面向对象的用电信息数据交换协议 - 报批稿:20170802 前言: 排版 ...
    庭说阅读 10,937评论 6 13
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,644评论 18 139
  • 单机存储引擎就是哈希表、B树等数据结构在机械磁盘、SSD等持久化介质上的实现。单机存储系统是单机存储引擎的一种封装...
    olostin阅读 2,429评论 0 5
  • 一、源题QUESTION 1The instance abnormally terminates because ...
    猫猫_tomluo阅读 1,595评论 0 2
  • 自己是个内向的人,不喜欢表达,于是常常让人误解,
    宋世巍阅读 160评论 0 0