Unity的AssetPostprocessor之Model:深入解析与实用案例 1
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的更多相关文章
- Unity3D for VR 学习(9): Unity Shader 光照模型 (illumination model)
关于光照模型 所谓模型,一般是由学术算法发起, 经过大量实际数据验证而成的可靠公式 现在还记得2009年做TD-SCDMA移动通信算法的时候,曾经看过自由空间传播模型(Free space propa ...
- Unity之GPS定位(高德解析)
Unity之GPS定位 Unity之GPS定位(高德解析) 前言 开篇 Unity版本及使用插件 正题 创建场景 写脚本 把脚本挂载到场景中 打包发布场景 安装真机并且测试 代码中的==Key==怎么 ...
- Android之三种网络请求解析数据(最佳案例)
AsyncTask解析数据 AsyncTask主要用来更新UI线程,比较耗时的操作可以在AsyncTask中使用. AsyncTask是个抽象类,使用时需要继承这个类,然后调用execute()方法. ...
- 用DOM4J解析XML文件案例
用DOM4J解析XML文件案例,由于DOM4J不像JAXP属于JAVASE里,所以如果要使用DOM4J,则必须额外引入jar包,如图:
- (转载)Android之三种网络请求解析数据(最佳案例)
[置顶] Android之三种网络请求解析数据(最佳案例) 2016-07-25 18:02 4725人阅读 评论(0) 收藏 举报 分类: Gson.Gson解析(1) 版权声明:本文为博主原创 ...
- DNS解析综合学习案例
DNS解析综合学习案例 #图右侧为做题前环境配置 #命令为红色 #命令加载内容为绿色 #vi编辑内容为蓝色 1.用户需把/dev/myvg/mylv逻辑卷以支持磁盘配额的方式挂载到网页目录下 [roo ...
- DNS解析综合学习案例(附详细答案)
1.用户需把/dev/myvg/mylv逻辑卷以支持磁盘配额的方式挂载到网页目录下2.在网页目录下创建测试文件index.html,内容为用户名称,通过浏览器访问测试3.创建用户账户,对LVM配置磁盘 ...
- .NET 用 Unity 依赖注入——概述注册和解析类型(1)
本文内容 Unity 概述 环境 一个真实的例子 类型注册(Type Registrations) 解析类型(Resolving Types) 跳槽,新公司使用了 Unity,初步看了一下,公司的使用 ...
- 《Unity系列》Json文件格式的解析——初级教程
Json作为轻量级的数据交换格式,被广泛应用于网络数据传输中.相关的解析工具层出不穷,一般掌握一个工具的应用其他的相应工具就能立马学会. 这里以C#中的LitJson为例给大家示范一下解析工具的用法. ...
- 【转】关于Unity协同程序(Coroutine)的全面解析
http://www.unity.5helpyou.com/2658.html 本篇文章我们学习下unity3d中协程Coroutine的的原理及使用 1.什么是协调程序 unity协程是一个能暂停执 ...
随机推荐
- Wolai 使用教程:嵌入小组件库,打造精美、强大的知识库主页
Wolai /我来云笔记在 2022.7.11 日的更新中,支持嵌入包括 NotionPet.芦笋.Replit 等在内的第三方应用.感谢 Wolai 云笔记官方对于 NotionPet 的支持. 趁 ...
- 超实用的Go语言基础教程,让你快速上手刷题!!
背景 工欲善其事,必先利其器.掌握Go的基础语法还不够,还需要勤加练习,修习"外功",才能达到出奇制胜的效果. 在大致了解Go语言的基本语法后,我就迫不得已地想使用这门语言.可是我 ...
- 2021-08-17:谷歌面试题扩展版,面值为1~N的牌组成一组,每次你从组里等概率的抽出1~N中的一张,下次抽会换一个新的组,有无限组,当累加和<a时,你将一直抽牌,当累加和>=a且<b时,你将获胜
2021-08-17:谷歌面试题扩展版,面值为1N的牌组成一组,每次你从组里等概率的抽出1N中的一张,下次抽会换一个新的组,有无限组,当累加和<a时,你将一直抽牌,当累加和>=a且< ...
- Java 世界的法外狂徒:反射
概述 反射(Reflection)机制是指在运行时动态地获取类的信息以及操作类的成员(字段.方法.构造函数等)的能力.通过反射,我们可以在编译时期未知具体类型的情况下,通过运行时的动态查找和调用. 虽 ...
- 实用的windows快捷键
Alt+F4 关闭窗口 win+D 显示桌面 win+Tab 切换窗口 Alt+Tab 应用之间的切换 win+E 打开我的电脑 Ctrl+Shift+Esc 打开任务管理器 Home 回到行首 En ...
- Python 九九乘法表的多种实现方式
简介 九九乘法表是初学者学习编程的必要练手题目之一,因此各种语言都有对应的实现方式,而 Python 也不例外.在 Python 中,我们可以使用多种方式来生成一个简单的九九乘法表. 本文共介绍了七种 ...
- Go语言中的结构体:灵活性与可扩展性的重要角色
1. 引言 结构体是Go语言中重要且灵活的概念之一.结构体的使用使得我们可以定义自己的数据类型,并将不同类型的字段组合在一起,实现更灵活的数据结构.本文旨在深入介绍Go语言中的结构体,揭示其重要性和灵 ...
- 10. docker方式下的mysql设置主从复制(一主两从)
上一篇 [centos 使用 docker 方式安装 mysql] 笔记中,我们在三个虚拟机中使用 docker 方式新建了三个 mysql 容器服务,那么我们这篇文章来记录下,如何在这三台机器中设置 ...
- 7. 特殊SQL的执行
1. 模糊查询 演示代码: /** * 测试模糊查询 * @param mohu * @return */ List<User> testMohu(@Param("mohu& ...
- Dlang 并行化
Dlang 并行化 好难受,dlang 生态太差,没办法,学了半天才明白. 我尽量以精炼的语言解释. 采用 定义,例子(代码),解释 的步骤讲解. 所以你可能看到很多代码,一点解释-- 我会省略一些 ...