Unity實(shí)現(xiàn)角色受擊身體邊緣發(fā)光特效
游戲中經(jīng)常需要制作角色受擊打的身體邊緣光效果,本文使用的方法是,給Renderer疊加一個制作好的邊緣光材質(zhì)球,并通過腳本動態(tài)控制邊緣光的漸變效果,表現(xiàn)出受擊后的邊緣光效果
工程結(jié)構(gòu)如下
1 創(chuàng)建一個材質(zhì)球HittedMatEffect.mat放在Assets/Resources/Material目錄中,使用TransparentRim.shader
注意代碼中用了Resources.Load,所以必須放在這個目錄里,你可以改成別的方式
2 場景中創(chuàng)建一個Sphere(球體),掛上Runner腳本,運(yùn)行,點(diǎn)擊屏幕任意位置,球體就會表現(xiàn)出受擊的邊緣光效果了
運(yùn)行效果
代碼
TransparentRim.shader
Shader "Effect/TransparentRim" { Properties{ _RimColor("Rim Color", Color) = (0.5,0.5,0.5,0.5) _InnerColor("Inner Color", Color) = (0.5,0.5,0.5,0.5) _InnerColorPower("Inner Color Power", Range(0.0,1.0)) = 0.5 _RimPower("Rim Power", Range(0.0,5.0)) = 2.5 _AlphaPower("Alpha Rim Power", Range(0.0,8.0)) = 4.0 _AllPower("All Power", Range(0.0, 10.0)) = 1.0 } SubShader{ Tags { "Queue" = "Transparent" } CGPROGRAM #pragma surface surf Lambert alpha struct Input { float3 viewDir; INTERNAL_DATA }; float4 _RimColor; float _RimPower; float _AlphaPower; float _AlphaMin; float _InnerColorPower; float _AllPower; float4 _InnerColor; void surf(Input IN, inout SurfaceOutput o) { half rim = 1.0 - saturate(dot(normalize(IN.viewDir), o.Normal)); o.Emission = _RimColor.rgb * pow(rim, _RimPower)*_AllPower + (_InnerColor.rgb * 2 * _InnerColorPower); o.Alpha = (pow(rim, _AlphaPower))*_AllPower; } ENDCG } Fallback "VertexLit" }
HittedMatEffect.cs
// HittedMatEffect.cs using UnityEngine; using System.Collections; public class HittedMatEffect : MonoBehaviour { bool mbActive = false; bool mbInit = false; Material mMat = null; public float mLife; private static int s_InnerColor = -1; private static int s_AllPower = -1; private static int s_AlphaPower = -1; void Awake() { s_InnerColor = Shader.PropertyToID("_InnerColor"); s_AllPower = Shader.PropertyToID("_AllPower"); s_AlphaPower = Shader.PropertyToID("_AlphaPower"); } // Use this for initialization /// <summary> /// 設(shè)置材質(zhì)顏色 /// </summary> /// <param name="color"></param> public void SetColor(Color color) { mMat.SetColor(s_InnerColor, color); } public void SetLifeTime(float time) { mLife = time; } public void Active() { if (!mbInit) AddEffect(); mMat.SetFloat(s_AllPower, 0.9f); mbActive = true; mLife = 0.2f; } void Update() { if (!mbActive) return; mLife -= Time.deltaTime; if (mLife < 0) { mbActive = false; mMat.SetFloat(s_AllPower, 0); } float v = Mathf.Sin((1 - mLife) * 8 * Mathf.PI) + 2; mMat.SetFloat(s_AlphaPower, v); } void AddEffect() { Object mat = Resources.Load("Material/HittedMatEffect"); mMat = GameObject.Instantiate(mat) as Material; foreach (var curMeshRender in transform.GetComponentsInChildren<Renderer>()) { Material[] newMaterialArray = new Material[curMeshRender.materials.Length + 1]; for (int i = 0; i < curMeshRender.materials.Length; i++) { if (curMeshRender.materials[i].name.Contains("HittedMatEffect")) { return; } else { newMaterialArray[i] = curMeshRender.materials[i]; } } if (null != mMat) newMaterialArray[curMeshRender.materials.Length] = mMat; curMeshRender.materials = newMaterialArray; } mbInit = true; } void RemoveEffect() { foreach (var curMeshRender in transform.GetComponentsInChildren<Renderer>()) { int newMaterialArrayCount = 0; for (int i = 0; i < curMeshRender.materials.Length; i++) { if (curMeshRender.materials[i].name.Contains("HittedMatEffect")) { newMaterialArrayCount++; } } if (newMaterialArrayCount > 0) { Material[] newMaterialArray = new Material[newMaterialArrayCount]; int curMaterialIndex = 0; for (int i = 0; i < curMeshRender.materials.Length; i++) { if (curMaterialIndex >= newMaterialArrayCount) { break; } if (!curMeshRender.materials[i].name.Contains("HittedMatEffect")) { newMaterialArray[curMaterialIndex] = curMeshRender.materials[i]; curMaterialIndex++; } } curMeshRender.materials = newMaterialArray; } } } }
Runner.cs
using System.Collections; using System.Collections.Generic; using UnityEngine; public class Runner : MonoBehaviour { // Use this for initialization void Start() { } // Update is called once per frame void Update() { if (Input.GetMouseButtonDown(0)) { HittedMatEffect sc = gameObject.GetComponent<HittedMatEffect>(); if (null == sc) sc = gameObject.AddComponent<HittedMatEffect>(); sc.Active(); sc.SetColor(Color.red); } } }
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Unity TextMeshPro實(shí)現(xiàn)富文本超鏈接默認(rèn)字體追加字體
這篇文章主要為大家介紹了Unity TextMeshPro實(shí)現(xiàn)富文本超鏈接默認(rèn)字體追加字體示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-01-01C#獲取鼠標(biāo)在listview右鍵點(diǎn)擊單元格的內(nèi)容方法
下面小編就為大家?guī)硪黄狢#獲取鼠標(biāo)在listview右鍵點(diǎn)擊單元格的內(nèi)容方法。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-01-01同步調(diào)用和異步調(diào)用WebService
本文給大家介紹webservice同步調(diào)用和異步調(diào)用,同步調(diào)用就是一個同步操作會阻塞整個當(dāng)前的進(jìn)程,直到這個操作完成才能執(zhí)行下一段代碼,異步調(diào)用不會阻塞啟動操作的調(diào)用線程,調(diào)用程序必須通過輪流檢測,或者等待完成信號來發(fā)現(xiàn)調(diào)用的完成。小伙伴們跟著小編一起學(xué)習(xí)2015-09-09深入多線程之:用Wait與Pulse模擬一些同步構(gòu)造的應(yīng)用詳解
本篇文章是對用Wait與Pulse模擬一些同步構(gòu)造的應(yīng)用進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05C# 獲取打印機(jī)當(dāng)前狀態(tài)的方法
C# 獲取打印機(jī)當(dāng)前狀態(tài)的方法,需要的朋友可以參考一下2013-04-04