Unity的AssetPostprocessor之Model:深入解析与实用案例 1

字节游侠
• 阅读 605

Unity AssetPostprocessor模型相关函数详解

在Unity中,AssetPostprocessor是一个非常有用的工具,它可以在导入资源时自动执行一些操作。在本文中,我们将重点介绍AssetPostprocessor中与模型相关的函数,并提供多个使用例子。

OnPostprocessModel

OnPostprocessModel是AssetPostprocessor中与模型相关的主要函数。它在导入模型时自动调用,并允许我们对模型进行一些自定义操作。下面是一个简单的例子:

using UnityEngine;
using UnityEditor;

public class MyModelPostprocessor : AssetPostprocessor
{
    void OnPostprocessModel(GameObject model)
    {
        // 在这里对模型进行自定义操作
    }
}

在这个例子中,我们创建了名为MyModelPostprocessor的AssetPostprocessor类,并重写了OnPostprocessModel函数。在这个函数中,我们可以对导入的模型进行自定义操作。

下面是一些常见的用:

1. 修改模型的材质

void OnPostprocessModel(GameObject model)
{
    Renderer[] renderers = model.GetComponentsInChildren<Renderer>();
    foreach (Renderer renderer in renderers)
    {
        Material[] materials = renderer.sharedMaterials;
        for (int i = 0; i < materials.Length; i++)
        {
            // 修改材质
            materials[i] = new Material(Shader.Find("Standard"));
        }
        renderer.sharedMaterials = materials;
    }
}

在这个例子中,我们获取了模型中所有的Renderer组件,并遍历每个Renderer的材质。然后,我们将每个材质替换为一个新的Standard材质。

2. 修改模型的网格

void OnPostprocessModel(GameObject model)
{
    MeshFilter[] meshFilters = model.GetComponentsInChildren<MeshFilter>();
    foreach (MeshFilter meshFilter in meshFilters)
    {
        // 修改网格
        Mesh mesh = meshFilter.sharedMesh;
        Vector3[] vertices = mesh.vertices;
        for (int i = 0; i < vertices.Length; i++)
        {
            vertices[i] += Vector3.up;
        }
        mesh.vertices = vertices;
        mesh.RecalculateNormals();
    }
}

在这个例子中,我们获取了模型中所有的MeshFilter组件,并遍历每个MeshFilter的网格。然后,我们将每个网格的顶点向上移动一个单位。

3. 修改模型的Transform

void OnPostprocessModel(GameObject model)
{
    model.transform.localScale = Vector3.one * 2;
    model.transform.position = Vector3.zero;
    model.transform.rotation = Quaternion.identity;
}

在这个例子中,我们直接修改了模型的Transform组件,将其缩放为原来的两倍,移动到原点,旋转为默认的旋转。

OnPreprocessModel

OnPreprocessModel是AssetPostprocessor中与模型相关的另一个函数。它在导入模型之前自动调用,并允许我们在导入之前对模型进行一些自定义操作。下面是一个简单的例子:

using UnityEngine;
using UnityEditor;

public class MyModelPostprocessor : AssetPostprocessor
{
    void OnPreprocessModel()
    {
        // 在这里对模型进行自定义操作
    }
}

在这个例子中,我们创建了一个名为MyModelPostprocessor的Assetprocessor类,并重写了OnPreprocessModel函数。在这个函数中,我们可以在导入模型之前对模型进行自定义操作。

下面是一些常见的用例:

1. 修改模型的导入设置

void OnPreprocessModel()
{
    ModelImporter importer =Importer as ModelImporter;
    importer.importMaterials = false;
    importer.importAnimation = false;
    importer.importTangents = ModelImporterTangents.None;
}

在这个例子中,我们获取了ModelImporter对象,并修改了导入模型的一些设置,例如不导入材质、动画和切线。

2. 修改模型的导入路径

void OnPreprocess()
{
    ModelImporter importer = assetImporter as ModelImporter;
    importer.importedTakeInfos[0].name = "MyAnimation";
    importer.animationType = ModelImporterAnimationType.Generic;
    importer.animationCompression = ModelImporterAnimationCompression.KeyframeReductionAndCompression;
    importer.animationPositionError = 0.01f;
    importer.animationRotationError = 0.01f;
    importer.animationScaleError = 0.01f;
    importer.animationWrapMode = WrapMode.Loop;
    importer.clipAnimations = new ModelImporterClipAnimation[]
    {
        new ModelImporterClipAnimation
        {
            name = "MyAnimation",
            firstFrame = 0,
            lastFrame = 100,
            loopTime = true,
            takeName = "MyAnimation",
        }
    };
    importer.clipAnimations[0].name = "MyAnimation";
    importer.clipAnimations[0].firstFrame = 0;
    importer.clipAnimations[0].lastFrame = 100;
    importer.clipAnimations[0].loopTime = true;
    importer.clipAnimations[0].takeName = "MyAnimation";
    importer.clipAnimations[0].wrapMode = WrapMode.Loop;
    importer.clipAnimations[0].lockRootRotation = true;
    importer.clipAnimations[0].lockRootHeightY = true;
    importer.clipAnimations[0].lockRootPositionXZ = true;
    importer.clipAnimations[0].curves = new AnimationClipCurveData[]
    {
        new AnimationClipCurveData
        {
            path = "MyObject",
            propertyName = "m_LocalPosition.x",
            curve = new AnimationCurve(new Keyframe[]
            {
                new Keyframe(0, 0),
                new Keyframe(1, 1),
                new Keyframe(2, 0),
            }),
        }
    };
    importer.clipAnimations[0].events = new AnimationEvent[]
    {
        new AnimationEvent
        {
            time = 1,
            functionName = "MyFunction",
            stringParameter = "MyParameter",
        }
    };
    importer.clipAnimations[0].maskType = ClipAnimationMaskType.CopyFromOther;
    importer.clipAnimations[0].maskSource = "MyOtherAnimation";
    importer.clipAnimations[0].maskSourceInstance = importer;
    importer.clipAnimations[0].maskBlendType = ClipAnimationMaskBlendType.Additive;
    importer.clipAnimations[0].maskNeedsUpdating = true;
    importer.clipAnimations[0].lockCurves = new bool[]
    {
        true,
        false,
        true,
    };
    importer.clipAnimations[0].loopPose = true;
    importer.clipAnimations[0].loopBlend = true;
    importer.clipAnimations[0].cycleOffset = 0.5f;
    importer.clipAnimations[0].loopBlendOrientation = true;
    importer.clipAnimations[0].loopBlendPositionY = true;
    importer.clipAnimations[0].loopBlendPositionXZ = true;
    importer.clipAnimations[0].keepOriginalOrientation = true;
    importer.clipAnimations[0].keepOriginalPositionY = true;
    importer.clipAnimations[0].keepOriginalPositionXZ = true;
    importer.clipAnimations[0].heightFromFeet = true;
    importer.clipAnimations[0].mirror = true;
    importer.clipAnimations[0].mirrorParameterCurveNames = new string[]
    {
        "MyParameter",
    };
    importer.clipAnimations[0].lockRootRotationX = true;
    importer.clipAnimations[0].lockRootRotationY = true;
    importer.clipAnimations[0].lockRootRotationZ = true;
    importer.clipAnimations[0].lockRootHeightY = true;
    importer.clipAnimations[0].lockRootPositionXZ = true;
    importer.clipAnimations[0].lockRootPositionY = true;
    importer.clipAnimations[0].curves = new AnimationClipCurveData[]
    {
        new AnimationClipCurveData
        {
            path = "MyObject",
            propertyName = "m_LocalPosition.x",
            curve = new AnimationCurve(new Keyframe[]
            {
                new Keyframe(0, 0),
                new Keyframe(1, 1),
                new Keyframe(2, 0),
            }),
        }
    };
}

在这个例子中,我们获取ModelImporter对象,并修改了导入模型的路径和一些动画设置,例如动画名称、循环模式、曲线和事件。

OnPostprocessGameObjectWithUserProperties

OnPostprocessGameObjectWithUserProperties是AssetPostprocessor中与用户自定义属性相关的函数。它在导入带有用户自定义属性的游戏对象时自动调用,并允许我们对游戏对象进行一些自定义操作。下面是一个简单的例子:

using UnityEngine;
using UnityEditor;

public class MyGameObjectPostprocessor : AssetPostprocessor
{
    void OnPostprocessGameObjectWithUserProperties(GameObject gameObject, string[] propNames, object[] values)
    {
        // 在这里对游戏对象进行自定义操作
    }
}

在这个例子中,我们创建了一个名为MyGameObjectPostprocessor的AssetPostprocessor类,并重写了OnPostprocessGameObjectWithUserProperties函数。在这个函数中,我们可以对导入的游戏对象进行自定义操作。

下面是一个常见的用例:

1. 修改游戏对象的材质

void OnPostprocessGameObjectWithUserProperties(GameObject gameObject, string[] propNames, object[] values)
{
    Renderer[] renderers = gameObject.GetComponentsInChildren<Renderer>();
    foreach (Renderer renderer in renderers)
    {
        Material[] materials = renderer.sharedMaterials;
        for (int i = 0; i < materials.Length; i++)
        {
            // 修改材质
            materials[i] = new Material(Shader.Find("Standard"));
        }
        renderer.sharedMaterials = materials;
    }
}

在这个例子中,我们获取了游戏对象中所有的Renderer组件,并遍历每个Renderer的材质。然后,我们将每个材质替换为一个新的Standard材质。

总结

在本文中,我们介绍了AssetPostprocessor中与模型相关的函数,并提供了多个使用例子。通过使用这些函数,我们可以导入模型时自动执行一些自定义操作,从而提高工作效率。

点赞
收藏
评论区
推荐文章
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
Wesley13 Wesley13
3年前
Unity C# Job System介绍 (一)
!(https://oscimg.oschina.net/oscnet/8d41bd52f45c4406b2f9b2143c372f369ee.jpg)一般情况下我们说unity不支持多线程,是因为我们只能在unity主线程内无限制的访问unity相关api。但是我们也是可以用Thread写多线程,只是在这样的分线程中我们无法获取unity组件运
Wesley13 Wesley13
3年前
Unity之SDK接入(Unity与Android通信)
首先介绍一点关于Android与unity通信的知识:完成通信主要靠unity中的class.jar包(在unity的安装目录下)。在unity中调用android的方法:jo.call("方法名"\,参数名\)       其中\\代表可有可无。在endroid中调用unityu的方法:导入class.jar包,继承
Wesley13 Wesley13
3年前
Unity Asset Bundle 角色换装示例 官方例子
《Unity官方案例精讲》一书中在第9章介绍了角色换装示例与AssetBundle的使用。示例中还展示了如何用SkinnedMeshRenderer渲染模型。虽然是好几年的示例了,但其中的换装的一些基本思想还是可以借鉴的,示例中动画的播放不用太在意,是legacy动画播放,了解下就好。下面先来看一下运行图。!character
Stella981 Stella981
3年前
ElasticSearch Client详解
本文将重点探讨ElasticSearchClient的相关知识,主要关注TransportClient与RestClient。Elasticsearchclient是我们进入ElastiSearch的大门。1、概述本文关注Elasticsearch与Java相关的API实现。所有Elasticsearch操作都是使用Client对象执行的。
Wesley13 Wesley13
3年前
Unity Inspector添加自定义按钮(Button)
     在Unity开发游戏的时候,为了有一个更快更方便的工作流,我们往往会在Editor下开发一些方便实用的工具。在工具中,用到最多,最关键的就是按钮,它是工具的首席执行官。下面就用最简单的代码来演示添加一个自定义按钮到Inspector当中。        案例:指定坐标后克隆新物体到场景。   1、 在UnityAssets下
Stella981 Stella981
3年前
Android so注入(inject)和Hook技术学习(二)——Got表hook之导入表hook
  全局符号表(GOT表)hook实际是通过解析SO文件,将待hook函数在got表的地址替换为自己函数的入口地址,这样目标进程每次调用待hook函数时,实际上是执行了我们自己的函数。  GOT表其实包含了导入表和导出表,导出表指将当前动态库的一些函数符号保留,供外部调用,导入表中的函数实际是在该动态库中调用外部的导出函数。  这里有几个关键点要
Wesley13 Wesley13
3年前
Unity中Awake与Start函数的调用情况总结
在Unity中编写脚本时,有一系列的可重写(override)函数供我们使用,其中的Awake与Start两个函数作为初始化与设置之用,几乎在每个脚本中都要用到。因此,正确的把握这两个函数的调用时机,就能让我们在程序开发过程中避免一些错误,提高开发效率。比较懒,所以就没有上图,欢迎大家的批评指正:)1、 Awake函数首先,我们来看un
Wesley13 Wesley13
3年前
Unity 资源导入时预处理 AssetPostprocessor
官方API: https://docs.unity3d.com/ScriptReference/AssetPostprocessor.html(https://www.oschina.net/action/GoToLink?urlhttps%3A%2F%2Fdocs.unity3d.com%2FScriptReference%2FAssetPostpr
liam liam
1年前
全面解读 Axios 的 GET 请求:最佳实践
在进行网络请求时,是一个非常常用的请求库。本文将介绍如何使用axios发起GET请求,并详细列出传参的几种写法。同时会提供一个实践案例,其中包含基本路由与请求处理的过程,并确保在IDE编辑器中可以顺利运行。什么是axios的GET请求?在开始之前,让我们简
小万哥 小万哥
1年前
Go 语言函数、参数和返回值详解
函数是一组语句,可以在程序中重复使用。函数不会在页面加载时自动执行。函数将通过调用函数来执行。创建函数要创建(通常称为声明)一个函数,请执行以下操作:使用func关键字。指定函数的名称,后跟括号()。最后,在花括号内添加定义函数应执行的代码。语法Gofun