Unity 编辑器预制体工具类PrefabUtility 常用函数和用法

Unity 编辑器预制体工具类PrefabUtility 常用函数和用法

简介

在Unity中,预制体(Prefab)是一种非常有用的工具,它允许我们创建可重复使用的对象和场景元素。Unity提供了许多内置的工具和函数来处理预制体,其中一个重要的类就是PrefabUtility。PrefabUtility类提供了一系列函数,用于创建、实例化和管理预制体。在本文中,我们将介绍PrefabUtility类的常用函数和用法。

创建和实例化

CreatePrefab

函数原型:public static GameObject CreatePrefab(string path, GameObject go);

CreatePrefab函数用于创建一个新的预制体。它接受两个参数:路径(path)和游戏对象(go)。路径参数指定了预制体的保存位置,而游戏对象参数则是要创建预制体的对象。

以下是CreatePrefab函数的示例使用代码:

using UnityEditor;
using UnityEngine;

public class PrefabCreator : MonoBehaviour
{
    [MenuItem("Tools/Create Prefab")]
    public static void CreatePrefab()
    {
        GameObject selectedObject = Selection.activeGameObject;
        if (selectedObject != null)
        {
            string path = "Assets/Prefabs/" + selectedObject.name + ".prefab";
            GameObject prefab = PrefabUtility.CreatePrefab(path, selectedObject);
            Debug.Log("Prefab created at " + path);
        }
    }
}

上述代码创建了一个名为PrefabCreator的脚本,并在Unity编辑器的菜单栏中添加了一个名为"Tools/Create Prefab"的选项。当用户选择一个游戏对象并点击该选项时,脚本将使用PrefabUtility.CreatePrefab函数创建一个预制体,并将其保存在Assets/Prefabs目录下。

CreateEmptyPrefab

函数原型:public static GameObject CreateEmptyPrefab(string path);

CreateEmptyPrefab函数用于创建一个空的预制体。它接受一个路径参数,指定了预制体的保存位置。

以下是CreateEmptyPrefab函数的示例使用代码:

using UnityEditor;
using UnityEngine;

public class EmptyPrefabCreator : MonoBehaviour
{
    [MenuItem("Tools/Create Empty Prefab")]
    public static void CreateEmptyPrefab()
    {
        string path = "Assets/Prefabs/EmptyPrefab.prefab";
        GameObject prefab = PrefabUtility.CreateEmptyPrefab(path);
        Debug.Log("Empty prefab created at " + path);
    }
}

上述代码创建了一个名为EmptyPrefabCreator的脚本,并在Unity编辑器的菜单栏中添加了一个名为"Tools/Create Empty Prefab"的选项。当用户点击该选项时,脚本将使用PrefabUtility.CreateEmptyPrefab函数创建一个空的预制体,并将其保存在Assets/Prefabs目录下。

InstantiatePrefab

函数原型:public static GameObject InstantiatePrefab(GameObject prefab, Transform parent);

InstantiatePrefab函数用于实例化一个预制体。它接受两个参数:预制体(prefab)和父级变换(parent)。预制体参数指定了要实例化的预制体,而父级变换参数指定了实例化后的对象的父级。

以下是InstantiatePrefab函数的示例使用代码:

using UnityEditor;
using UnityEngine;

public class PrefabInstantiator : MonoBehaviour
{
    [MenuItem("Tools/Instantiate Prefab")]
    public static void InstantiatePrefab()
    {
        GameObject prefab = AssetDatabase.LoadAssetAtPath<GameObject>("Assets/Prefabs/MyPrefab.prefab");
        Transform parent = GameObject.Find("Parent").transform;
        GameObject instance = PrefabUtility.InstantiatePrefab(prefab, parent) as GameObject;
        Debug.Log("Prefab instantiated");
    }
}

上述代码创建了一个名为PrefabInstantiator的脚本,并在Unity编辑器的菜单栏中添加了一个名为"Tools/Instantiate Prefab"的选项。当用户点击该选项时,脚本将使用PrefabUtility.InstantiatePrefab函数实例化一个预制体,并将其作为子对象添加到名为"Parent"的游戏对象下。

InstantiatePrefabInScene

函数原型:public static GameObject InstantiatePrefabInScene(GameObject prefab, Vector3 position, Quaternion rotation);

InstantiatePrefabInScene函数用于在场景中实例化一个预制体。它接受三个参数:预制体(prefab)、位置(position)和旋转(rotation)。预制体参数指定了要实例化的预制体,位置参数指定了实例化后的对象的位置,旋转参数指定了实例化后的对象的旋转。

以下是InstantiatePrefabInScene函数的示例使用代码:

using UnityEditor;
using UnityEngine;

public class ScenePrefabInstantiator : MonoBehaviour
{
    [MenuItem("Tools/Instantiate Prefab in Scene")]
    public static void InstantiatePrefabInScene()
    {
        GameObject prefab = AssetDatabase.LoadAssetAtPath<GameObject>("Assets/Prefabs/MyPrefab.prefab");
        Vector3 position = new Vector3(0, 0, 0);
        Quaternion rotation = Quaternion.identity;
        GameObject instance = PrefabUtility.InstantiatePrefabInScene(prefab, position, rotation) as GameObject;
        Debug.Log("Prefab instantiated in scene");
    }
}

上述代码创建了一个名为ScenePrefabInstantiator的脚本,并在Unity编辑器的菜单栏中添加了一个名为"Tools/Instantiate Prefab in Scene"的选项。当用户点击该选项时,脚本将使用PrefabUtility.InstantiatePrefabInScene函数在场景中实例化一个预制体,并将其放置在位置(0, 0, 0),并保持旋转为默认值。

操作和修改函数

ReplacePrefab

函数签名:public static GameObject ReplacePrefab(GameObject go, GameObject prefab, ReplacePrefabOptions options = ReplacePrefabOptions.Default);

该函数用于替换预制体的实例。它接受两个参数:go表示要替换的游戏对象实例,prefab表示要替换成的预制体。可选参数options用于指定替换预制体的选项。

使用示例:

GameObject go = GameObject.Find("MyGameObject");
GameObject prefab = Resources.Load<GameObject>("Prefabs/MyPrefab");
PrefabUtility.ReplacePrefab(go, prefab, ReplacePrefabOptions.ConnectToPrefab);

ConnectGameObjectToPrefab

函数签名:public static GameObject ConnectGameObjectToPrefab(GameObject go, GameObject prefab);

该函数用于将游戏对象连接到预制体。它接受两个参数:go表示要连接的游戏对象,prefab表示要连接到的预制体。

使用示例:

GameObject go = GameObject.Find("MyGameObject");
GameObject prefab = Resources.Load<GameObject>("Prefabs/MyPrefab");
PrefabUtility.ConnectGameObjectToPrefab(go, prefab);

DisconnectPrefabInstance

函数签名:public static void DisconnectPrefabInstance(GameObject gameObject);

该函数用于断开游戏对象与预制体的连接。它接受一个参数:gameObject表示要断开连接的游戏对象。

使用示例:

GameObject go = GameObject.Find("MyGameObject");
PrefabUtility.DisconnectPrefabInstance(go);

ApplyPrefabInstance

函数签名:public static void ApplyPrefabInstance(GameObject instanceRoot, InteractionMode interactionMode = InteractionMode.UserAction);

该函数用于将游戏对象的修改应用到预制体实例。它接受两个参数:instanceRoot表示要应用修改的游戏对象实例的根节点,interactionMode用于指定应用修改的交互模式。

使用示例:

GameObject go = GameObject.Find("MyGameObject");
PrefabUtility.ApplyPrefabInstance(go, InteractionMode.AutomatedAction);

RevertPrefabInstance

函数签名:public static void RevertPrefabInstance(GameObject gameObject);

该函数用于还原游戏对象到预制体实例的状态。它接受一个参数:gameObject表示要还原的游戏对象。

使用示例:

GameObject go = GameObject.Find("MyGameObject");
PrefabUtility.RevertPrefabInstance(go);

查询和检查函数

GetPrefabType

函数签名:public static PrefabType GetPrefabType(Object targetObject);

该函数用于获取预制体的类型。它接受一个参数targetObject,表示要查询的对象,可以是游戏对象或组件。

返回值类型为PrefabType,表示预制体的类型。可能的返回值包括:

  • None:表示对象不是预制体的一部分。
  • Prefab:表示对象是一个完整的预制体。
  • PrefabInstance:表示对象是一个预制体的实例。
  • DisconnectedPrefabInstance:表示对象是一个断开连接的预制体实例。
  • PrefabAsset:表示对象是一个预制体资源。

使用示例:

GameObject go = GameObject.Find("MyGameObject");
PrefabType prefabType = PrefabUtility.GetPrefabType(go);
Debug.Log("Prefab Type: " + prefabType);

GetPrefabParent

函数签名:public static GameObject GetPrefabParent(GameObject gameObject);

该函数用于获取预制体的父级。它接受一个参数gameObject,表示要查询的游戏对象。

返回值类型为GameObject,表示预制体的父级对象。如果对象不是预制体的一部分,则返回null

使用示例:

GameObject go = GameObject.Find("MyGameObject");
GameObject prefabParent = PrefabUtility.GetPrefabParent(go);
Debug.Log("Prefab Parent: " + prefabParent);

GetPrefabObject

函数签名:public static GameObject GetPrefabObject(GameObject gameObject);

该函数用于获取预制体的对象。它接受一个参数gameObject,表示要查询的游戏对象。

返回值类型为GameObject,表示预制体的对象。如果对象不是预制体的一部分,则返回null

使用示例:

GameObject go = GameObject.Find("MyGameObject");
GameObject prefabObject = PrefabUtility.GetPrefabObject(go);
Debug.Log("Prefab Object: " + prefabObject);

GetPrefabInstanceHandle

函数签名:public static PrefabInstanceHandle GetPrefabInstanceHandle(GameObject gameObject);

该函数用于获取预制体实例的句柄。它接受一个参数gameObject,表示要查询的游戏对象。

返回值类型为PrefabInstanceHandle,表示预制体实例的句柄。如果对象不是预制体的实例,则返回一个无效的句柄。

使用示例:

GameObject go = GameObject.Find("MyGameObject");
PrefabInstanceHandle instanceHandle = PrefabUtility.GetPrefabInstanceHandle(go);
Debug.Log("Prefab Instance Handle: " + instanceHandle);

GetCorrespondingObjectFromSource

函数签名:public static Object GetCorrespondingObjectFromSource(Object sourceObject);

该函数用于从预制体实例获取对应的源对象。它接受一个参数sourceObject,表示预制体实例中的对象。

返回值类型为Object,表示源对象。如果对象不是预制体实例的一部分,则返回null

使用示例:

GameObject go = GameObject.Find("MyGameObject");
Object sourceObject = PrefabUtility.GetCorrespondingObjectFromSource(go);
Debug.Log("Source Object: " + sourceObject);

其他类型

1. 获取预制体实例根节点的路径

函数名:GetPrefabAssetPathOfNearestInstanceRoot

public static string GetPrefabAssetPathOfNearestInstanceRoot(GameObject instanceRoot);

该函数用于获取最近的预制体实例根节点的路径。它接受一个GameObject参数,表示预制体实例的根节点,然后返回一个字符串,表示该预制体实例根节点所对应的预制体的路径。

返回值:

  • 如果传入的GameObject参数是一个预制体实例的根节点,则返回该预制体的路径。
  • 如果传入的GameObject参数不是预制体实例的根节点,则返回空字符串。

示例代码:

GameObject instance = PrefabUtility.GetPrefabInstanceHandle(gameObject);
string prefabPath = PrefabUtility.GetPrefabAssetPathOfNearestInstanceRoot(instance);
Debug.Log("Prefab Path: " + prefabPath);

2. 获取预制体实例的数量

函数名:GetPrefabInstanceCount

public static int GetPrefabInstanceCount(Object targetPrefab);

该函数用于获取指定预制体的实例数量。它接受一个Object参数,表示目标预制体,可以是预制体的引用或实例。

返回值:

  • 返回一个整数,表示指定预制体的实例数量。

示例代码:

GameObject prefab = PrefabUtility.GetCorrespondingObjectFromSource(gameObject);
int instanceCount = PrefabUtility.GetPrefabInstanceCount(prefab);
Debug.Log("Instance Count: " + instanceCount);

3. 获取预制体实例的列表

函数名:GetPrefabInstanceList

public static List<GameObject> GetPrefabInstanceList(Object targetPrefab);

该函数用于获取指定预制体的所有实例列表。它接受一个Object参数,表示目标预制体,可以是预制体的引用或实例。

返回值:

  • 返回一个List<GameObject>,包含指定预制体的所有实例。

示例代码:

GameObject prefab = PrefabUtility.GetCorrespondingObjectFromSource(gameObject);
List<GameObject> instanceList = PrefabUtility.GetPrefabInstanceList(prefab);
foreach (GameObject instance in instanceList)
{
    Debug.Log("Prefab Instance: " + instance.name);
}

4. 获取预制体实例的状态

函数名:GetPrefabInstanceStatus

public static PrefabInstanceStatus GetPrefabInstanceStatus(Object targetObject);

该函数用于获取指定对象的预制体实例状态。它接受一个Object参数,表示目标对象,可以是预制体的引用或实例。

返回值:

  • 返回一个PrefabInstanceStatus枚举值,表示指定对象的预制体实例状态。可能的枚举值包括:
    • Connected:表示对象是一个预制体实例,并且与预制体保持连接。
    • Disconnected:表示对象是一个预制体实例,但与预制体断开连接。
    • NotAPrefab:表示对象不是一个预制体实例。

示例代码:

GameObject instance = PrefabUtility.GetPrefabInstanceHandle(gameObject);
PrefabInstanceStatus instanceStatus = PrefabUtility.GetPrefabInstanceStatus(instance);
Debug.Log("Instance Status: " + instanceStatus);

结论

在本文中,我们介绍了Unity编辑器中PrefabUtility类的常用函数和用法。我们学习了如何获取预制体实例的路径、数量、列表和状态,并解释了每个函数返回值的具体含义。通过使用PrefabUtility类,我们可以更好地管理和操作预制体,提高开发效率。

参考文档:Unity Script Reference - PrefabUtility

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

推荐阅读更多精彩内容