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-10-10:OOM都有哪些,说出几种?

    福哥答案2020-10-10:#福大大架构师每日一题# [答案参考了此链接:](https://cloud.tencent.com/developer/article/1480668) 本地方法栈:1 ...

  2. 2022-05-26:void add(int L, int R, int C)代表在arr[L...R]上每个数加C, int get(int L, int R)代表查询arr[L...R]上的累加

    2022-05-26:void add(int L, int R, int C)代表在arr[L-R]上每个数加C, int get(int L, int R)代表查询arr[L-R]上的累加和, 假 ...

  3. 2021-09-24:给定一个正整数 n ,输出的第 n 项。前五项如下:1:1。2:11。3:21。4:1211。5:111221。第一项是数字 1 。描述前一项,这个数是 1 即 “ 一 个 1

    2021-09-24:给定一个正整数 n ,输出的第 n 项.前五项如下:1:1.2:11.3:21.4:1211.5:111221.第一项是数字 1 .描述前一项,这个数是 1 即 " 一 ...

  4. 【GiraKoo】常用编码的对比(ASCII,GB2312,GBK,GB18030,UCS,Unicode)

    常用编码的对比(ASCII,GB2312,GBK,GB18030,UCS,Unicode) 在程序开发中,文字编码一直扮演着人畜无害,却背后捅一刀的角色. 可能在源代码文件中,注释莫名其妙地变成了乱码 ...

  5. nginx: [emerg] https protocol requires SSL support in /usr/local/nginx/conf/nginx.conf:50

    最近在nginx中配置一个443端口 一.安装nginx 首先得先安装个nginx 1.安装依赖包 # 一键安装上面四个依赖 [root@dex ~]# yum -y install gcc zlib ...

  6. https 原理与实践

    https 原理与实践 经典三问,是什么,为什么,怎么做? 是什么 是一种http的安全协议,在tcp ip网络模型里,http应用层是在tcp 传输层之上的,https协议规定了在tcp传输层之上还 ...

  7. .NET周报 【5月第4期 2023-05-27】

    国内文章 C#使用词嵌入向量与向量数据库为大语言模型(LLM)赋能长期记忆实现私域问答机器人落地之openai接口平替 https://www.cnblogs.com/gmmy/p/17430613. ...

  8. 3、数据库:Oracle部署 - 系统部署系列文章

    Oracle数据库的安装,以前写过一篇,这次将新版的安装再记录一次,让读者能够有所了解,笔者也能够记录下最新版的安装过程. 一.数据库下载: Oracle最新版目前在官网是19c,从下面这个链接进去下 ...

  9. Apikit SaaS 10.9.0 版本更新: 接口测试支持通过 URL 请求大型文件,支持导出为 Postman 格式文件

    Hi,大家好! Eolink Apikit 即将在 2023年 6月 8日晚 18:00 开始更新 10.9.0 版本.本次版本更新主要是对多个应用级资源合并,并基于此简化付费套餐和降低费率. 本次应 ...

  10. 【论文阅读】CYCADA CYCLE-CONSISTENT ADVERSARIAL DOMAIN ADAPTATION

    github code CyCADA论文中,定义了一种问题--无监督适配,即仅提供源数据 \(X_S\) 和源标签 \(Y_S\),以及源域目标数据\(X_T\),没有目标标签或者不利用它.问题的目的 ...