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中与模型相关的函数,并提供了多个使用例子。通过使用这些函数,我们可以导入模型时自动执行一些自定义操作,从而提高工作效率。

Unity的AssetPostprocessor之Model:深入解析与实用案例 1的更多相关文章

  1. Unity3D for VR 学习(9): Unity Shader 光照模型 (illumination model)

    关于光照模型 所谓模型,一般是由学术算法发起, 经过大量实际数据验证而成的可靠公式 现在还记得2009年做TD-SCDMA移动通信算法的时候,曾经看过自由空间传播模型(Free space propa ...

  2. Unity之GPS定位(高德解析)

    Unity之GPS定位 Unity之GPS定位(高德解析) 前言 开篇 Unity版本及使用插件 正题 创建场景 写脚本 把脚本挂载到场景中 打包发布场景 安装真机并且测试 代码中的==Key==怎么 ...

  3. Android之三种网络请求解析数据(最佳案例)

    AsyncTask解析数据 AsyncTask主要用来更新UI线程,比较耗时的操作可以在AsyncTask中使用. AsyncTask是个抽象类,使用时需要继承这个类,然后调用execute()方法. ...

  4. 用DOM4J解析XML文件案例

    用DOM4J解析XML文件案例,由于DOM4J不像JAXP属于JAVASE里,所以如果要使用DOM4J,则必须额外引入jar包,如图:

  5. (转载)Android之三种网络请求解析数据(最佳案例)

    [置顶] Android之三种网络请求解析数据(最佳案例) 2016-07-25 18:02 4725人阅读 评论(0) 收藏 举报  分类: Gson.Gson解析(1)  版权声明:本文为博主原创 ...

  6. DNS解析综合学习案例

    DNS解析综合学习案例 #图右侧为做题前环境配置 #命令为红色 #命令加载内容为绿色 #vi编辑内容为蓝色 1.用户需把/dev/myvg/mylv逻辑卷以支持磁盘配额的方式挂载到网页目录下 [roo ...

  7. DNS解析综合学习案例(附详细答案)

    1.用户需把/dev/myvg/mylv逻辑卷以支持磁盘配额的方式挂载到网页目录下2.在网页目录下创建测试文件index.html,内容为用户名称,通过浏览器访问测试3.创建用户账户,对LVM配置磁盘 ...

  8. .NET 用 Unity 依赖注入——概述注册和解析类型(1)

    本文内容 Unity 概述 环境 一个真实的例子 类型注册(Type Registrations) 解析类型(Resolving Types) 跳槽,新公司使用了 Unity,初步看了一下,公司的使用 ...

  9. 《Unity系列》Json文件格式的解析——初级教程

    Json作为轻量级的数据交换格式,被广泛应用于网络数据传输中.相关的解析工具层出不穷,一般掌握一个工具的应用其他的相应工具就能立马学会. 这里以C#中的LitJson为例给大家示范一下解析工具的用法. ...

  10. 【转】关于Unity协同程序(Coroutine)的全面解析

    http://www.unity.5helpyou.com/2658.html 本篇文章我们学习下unity3d中协程Coroutine的的原理及使用 1.什么是协调程序 unity协程是一个能暂停执 ...

随机推荐

  1. 2020-09-23:TCP头部信息有哪些?

    福哥答案2020-09-23:#福大大架构师每日一题# 福哥口诀法:T源目序缺首保 紧确推和复同终 窗校紧选数(TCP格式:源端口,目的端口,序号,确认号,首部长度,保留,紧急位URG,确认位ACK, ...

  2. 2020-10-29:使用redis实现分布式限流组件,要求高并发场景同一IP一分钟内只能访问100次,超过限制返回异常,写出实现思路或伪代码均可。

    福哥答案2020-10-29: 简单回答:固定窗口:string.key存ip,value存次数.滑动窗口:list.key存ip,value=list,存每次访问的时间. 中级回答:固定窗口:用re ...

  3. ModuleNotFoundError: No module named 'pyecharts'

    ModuleNotFoundError: No module named 'pyecharts' 解决: pip install pyecharts

  4. .Net8顶级技术:边界检查之IR解析(慎入)

    前言 C#这种语言之所以号称安全的,面向对象的语言.这个安全两个字可不是瞎叫的哦.因为JIT会检查任何可能超出分配范围的数值,以便使其保持在安全边界内.这里有两个概念,其一边界检查,其二IR解析.后者 ...

  5. 终于把 7 年前的 Docker Hub 账号恢复了

    折腾 docker,向 Docker Hub 提交镜像的时候发现原来自己在 2014 年就已经注册过 Docker Hub 的账号了,而且在 https://hub.docker.com/u/shen ...

  6. Java基础之基础语法与面向对象

    前言 小知识 Java由Sun公司于1995年推出,2009年Sun公司被Oracle公司收购,取得Java的版权 Java之父:James Gosling(詹姆斯·高斯林) 专业术语 JDK:jav ...

  7. 基于uniapp+vite4+vue3搭建跨端项目|uni-app+uview-plus模板

    最近得空学习了下uniapp结合vue3搭建跨端项目.之前也有使用uniapp开发过几款聊天/仿抖音/后台管理等项目,但都是基于vue2开发.随着vite.js破局出圈,越来越多的项目偏向于vue3开 ...

  8. 一次Mybaits查询的源码分析

    很好奇Mybaits是怎么将xml和mapper对应起来的,用一段比较简单的demo去debug追踪一下源码看看 先用xml配置的方式,看懂了再去看注解的方式是怎么实现的 获取Mapper Mybai ...

  9. Devexpress如何获取RadioGroup选中项的值和显示值

    分享一个小技巧,如题目所示,DEV控件如何获取RadioGroup选中项的值和显示值.也是在网上找了很久,看了大家都是通过SelectIndex的值定位选中的按钮,并没有说取选中项的值,所以自己研究了 ...

  10. Junit4 一直处于运行中的排查过程

    新买了一个Macbook Pro . 之前的工程搬家过来, 这天要跑个单元测试. 发现Junit4 一直处于运行中.没有错误信息,没有用例执行结果.遂开始排查原因. 这里插一句,苹果芯片的Mbp还是很 ...