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. 2022-02-19:安装栅栏。 在一个二维的花园中,有一些用 (x, y) 坐标表示的树。由于安装费用十分昂贵,你的任务是先用最短的绳子围起所有的树。只有当所有的树都被绳子包围时,花园才能围好栅栏。

    2022-02-19:安装栅栏. 在一个二维的花园中,有一些用 (x, y) 坐标表示的树.由于安装费用十分昂贵,你的任务是先用最短的绳子围起所有的树.只有当所有的树都被绳子包围时,花园才能围好栅栏. ...

  2. Django笔记四十之运行Django环境的python脚本

    本文首发于公众号:Hunter后端 原文链接:Django笔记四十之运行Django环境的python脚本 这一篇笔记介绍如何在 Django 中运行脚本. 假设说我们要实现一个功能,需要获取 blo ...

  3. FPGA输出时钟jitter

    If customer performs simple clock forwarding from clock source -> FPGA clock input -> FPGA clo ...

  4. java设计模式【抽象工厂模式】

    java设计模式[抽象工厂模式] 抽象工厂模式 抽象工厂模式是对简单工厂模式的一个变种,它允许通过一个统一的接口来创建不同的产品实例,而无需指定具体的子类.在这个模式中,我们只关心产品的抽象接口,而将 ...

  5. 贪心算法基础及leetcode例题

    参考 理论 本质:找到每个阶段的局部最优,然后去推导得到全局最优 两个极端:常识&&很难: 很多同学通过了贪心的题目,但都不知道自己用了贪心算法,因为贪心有时候就是常识性的推导,所以会 ...

  6. Python批量下载鼠标样式,自动化一条龙处理详解

    目录 前情提要 爬虫环境 抓包分析请求 编写代码: 展示效果 结束语 前情提要 最近发现一款特别好看的壁纸软件,其中提供了鼠标样式,感觉很好看!很精致!心想肯定是请求下载然后启用鼠标样式, 那么发送请 ...

  7. 最全的git操作命令(持续更新)

    当前使用git进行版本管理越来越频繁,但是难免还是有些命令记不全,曾当前闲暇记录一下,免得需要时漫天找寻 目录 一. 配置用户信息 1.git config [配置git 用户信息] 2.git co ...

  8. 《MS17-010(永恒之蓝)—漏洞复现及防范》

    作者: susususuao 免责声明:本文仅供学习研究,严禁从事非法活动,任何后果由使用者本人负责. 一. 什么是永恒之蓝? - 永恒之蓝 永恒之蓝(Eternal Blue)是一种利用Window ...

  9. [Docker] Docker之安装Nginx

    0 序言 略 1 安装步骤 Step1 下载镜像 搜素.下载镜像 https://hub.docker.com/_/nginx?tab=tags 这里选择官方镜像1.22.0版本 docker sea ...

  10. shell编程-文件归档

    需求说明:设置定时任务,每天凌晨1点进行将指定目录(/root/scripts)下文件按照archive_目录名_年月日.tar.gz的格式归档存放到/root/archive 路径下. 1.编写脚本 ...