Unity3D實(shí)現(xiàn)模型淡入淡出效果
開(kāi)發(fā)中我們不僅需要UI界面淡入淡出,有時(shí)候還需要模型淡入淡出。我們?cè)诿姘迳闲薷腸olor的a值時(shí)發(fā)現(xiàn)并沒(méi)有效果。那是因?yàn)槲覀冊(cè)O(shè)置的RenderingMode是Opaque。官方標(biāo)準(zhǔn)shader中的Opaque pass段是不能顯示半透明效果的,所以我們需要設(shè)置RenderingMode為Fade或者Transparent。然后在修改color的a值,達(dá)到淡入淡出的效果。效果如下:
通常我們淡入一個(gè)模型只會(huì)傳入這個(gè)模型的GameObject,所以我們自寫(xiě)一個(gè)類(lèi)來(lái)處理這個(gè)模型淡入的一些事件。代碼如下:
using System.Collections; using System.Collections.Generic; using UnityEngine; using DG.Tweening; public class FadeModel { private GameObject model;//傳入的模型 private float fadeTime = 2f;//默認(rèn)淡入時(shí)間為2s private List<Material> materials = new List<Material>(); public FadeModel(GameObject model,float fadeTime=2f) { this.model = model; this.fadeTime = fadeTime; MeshRenderer[] meshRenderers = model.GetComponentsInChildren<MeshRenderer>(); foreach(MeshRenderer mr in meshRenderers) { Material[] materals = mr.materials; foreach(Material m in materals) { if (!materials.Contains(m)) { materials.Add(m); } } } } //隱藏模型的淡隱效果 public void HideModel() { for(int i=0;i< materials.Count;i++) { Material m = materials[i]; Color color = m.color; m.color = new Color(color.r, color.g, color.b, 1);//這里一定要重新設(shè)置下Fade模式下的color a值 為1 不然 經(jīng)過(guò)一次顯示他會(huì)一直顯示為0 setMaterialRenderingMode(m,RenderingMode.Fade); m.DOColor(new Color(color.r, color.g, color.b, 0), fadeTime); } } //當(dāng)我們隱藏完后還需要設(shè)置回來(lái) 不然他下次顯示使用就是透明狀態(tài) public void ShowModel() { for (int i = 0; i < materials.Count; i++) { Material m = materials[i]; Color color = m.color; setMaterialRenderingMode(m, RenderingMode.Opaque); } } public enum RenderingMode { Opaque, Cutout, Fade, Transparent } //設(shè)置材質(zhì)的渲染模式 這段在我之前的博客有講 代碼設(shè)置渲染模式 private void setMaterialRenderingMode(Material material, RenderingMode renderingMode) { switch (renderingMode) { case RenderingMode.Opaque: material.SetInt("_SrcBlend", (int)UnityEngine.Rendering.BlendMode.One); material.SetInt("_DstBlend", (int)UnityEngine.Rendering.BlendMode.Zero); material.SetInt("_ZWrite", 1); material.DisableKeyword("_ALPHATEST_ON"); material.DisableKeyword("_ALPHABLEND_ON"); material.DisableKeyword("_ALPHAPREMULTIPLY_ON"); material.renderQueue = -1; break; case RenderingMode.Cutout: material.SetInt("_SrcBlend", (int)UnityEngine.Rendering.BlendMode.One); material.SetInt("_DstBlend", (int)UnityEngine.Rendering.BlendMode.Zero); material.SetInt("_ZWrite", 1); material.EnableKeyword("_ALPHATEST_ON"); material.DisableKeyword("_ALPHABLEND_ON"); material.DisableKeyword("_ALPHAPREMULTIPLY_ON"); material.renderQueue = 2450; break; case RenderingMode.Fade: material.SetInt("_SrcBlend", (int)UnityEngine.Rendering.BlendMode.SrcAlpha); material.SetInt("_DstBlend", (int)UnityEngine.Rendering.BlendMode.OneMinusSrcAlpha); material.SetInt("_ZWrite", 0); material.DisableKeyword("_ALPHATEST_ON"); material.EnableKeyword("_ALPHABLEND_ON"); material.DisableKeyword("_ALPHAPREMULTIPLY_ON"); material.renderQueue = 3000; //material.SetFloat("" _Mode & quot;", 2); break; case RenderingMode.Transparent: material.SetInt("_SrcBlend", (int)UnityEngine.Rendering.BlendMode.One); material.SetInt("_DstBlend", (int)UnityEngine.Rendering.BlendMode.OneMinusSrcAlpha); material.SetInt("_ZWrite", 0); material.DisableKeyword("_ALPHATEST_ON"); material.DisableKeyword("_ALPHABLEND_ON"); material.EnableKeyword("_ALPHAPREMULTIPLY_ON"); material.renderQueue = 3000; break; } } }
我們?cè)跇?gòu)造函數(shù)里傳入模型,然后我們獲取到這個(gè)模型下所有的材質(zhì)球添加到一個(gè)材質(zhì)球數(shù)組里用來(lái)管理控制。還要引入Dotween插件哦。我們是用的Dotween來(lái)做的淡出效果。然后我們就可以在我們需要的地方來(lái)調(diào)用淡出效果了。
public class Test : MonoBehaviour { public GameObject model; FadeModel fadeModel; // Use this for initialization void Start () { fadeModel = new FadeModel(model); fadeModel.HideModel(); } private void OnDisable() { fadeModel.ShowModel(); } }
在編輯器模式測(cè)試的時(shí)候,我們?cè)诮Y(jié)束測(cè)試的時(shí)候要改回來(lái)渲染模式,不然在結(jié)束編輯器運(yùn)行的時(shí)候模型顯示的是半透明模式。編輯器不給你還原之前的狀態(tài)。同理,在我們淡出模型之后要設(shè)置回來(lái)模型的渲染模式,不然在下次進(jìn)入該場(chǎng)景使用該模型的時(shí)候,模型的渲染方式是Fade就出現(xiàn)錯(cuò)誤啦。
我在用人物做這個(gè)淡出效果測(cè)試時(shí)發(fā)現(xiàn)效果很不好。半透明效果有時(shí)候會(huì)讓后面的辮子渲染到前面了。所以對(duì)于精細(xì)顯示的效果我們還是去求助我們的shader程序員吧,讓他寫(xiě)一個(gè)貼圖消融的shader。但是對(duì)于工業(yè)級(jí)別的粗糙顯示,我們這個(gè)方案還是完全夠用的。
還有一個(gè),我們?cè)讷@取一個(gè)物體下的所有組件時(shí),使用的代碼是:
MeshRenderer[] meshRenderers = model.GetComponentsInChildren<MeshRenderer>();
當(dāng)時(shí)當(dāng)我們model下的子物體setactive是false的時(shí)候是不能被找到的。如果我們也想找到這些物體,我們就需要在上面代碼括號(hào)里加一個(gè)true。如下:
MeshRenderer[] meshRenderers = model.GetComponentsInChildren<MeshRenderer>(true);
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
C# 通過(guò) inline-asm 解決嵌入x86匯編
此篇文章通過(guò)C#語(yǔ)言解決嵌入x86匯編,主要通過(guò)INline-asm方法來(lái)實(shí)現(xiàn),下面我通過(guò)圖片和代碼的形式給大家分享下,需要的朋友可以參考下2015-07-07C#中的應(yīng)用程序接口介紹及實(shí)現(xiàn),密封類(lèi)與密封方法
今天小編就為大家分享一篇關(guān)于C#中的應(yīng)用程序接口介紹及實(shí)現(xiàn),密封類(lèi)與密封方法,小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧2018-10-10WindowsForm實(shí)現(xiàn)TextBox占位符Placeholder提示功能
這篇文章主要介紹了WindowsForm實(shí)現(xiàn)TextBox占位符Placeholder提示,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-07-07在Winform框架界面中改變并存儲(chǔ)界面皮膚樣式的方法
下面小編就為大家分享一篇在Winform框架界面中改變并存儲(chǔ)界面皮膚樣式的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助2017-11-11http圖片上傳安全性問(wèn)題 根據(jù)ContentType (MIME) 判斷其實(shí)不準(zhǔn)確、不安全
圖片上傳常用的類(lèi)型判斷方法有這么幾種---截取擴(kuò)展名、獲取文件ContentType (MIME) 、讀取byte來(lái)判斷(這個(gè)什么叫法來(lái)著?)。下面由腳本之家小編跟大家分享圖片上傳安全性問(wèn)題,感興趣的朋友一起看看吧2015-09-09C#使用WinRar命令進(jìn)行壓縮和解壓縮操作的實(shí)現(xiàn)方法
這篇文章主要介紹了C#使用WinRar命令進(jìn)行壓縮和解壓縮操作的實(shí)現(xiàn)方法,涉及C#基于Process類(lèi)操作WinRar命令的相關(guān)實(shí)現(xiàn)技巧,代碼簡(jiǎn)潔實(shí)用,需要的朋友可以參考下2016-06-06