Unity实现角色受击身体边缘发光特效

游戏中经常需要制作角色受击打的身体边缘光效果,本文使用的方法是,给Renderer叠加一个制作好的边缘光材质球,并通过脚本动态控制边缘光的渐变效果,表现出受击后的边缘光效果

工程结构如下

1 创建一个材质球HittedMatEffect.mat放在Assets/Resources/Material目录中,使用TransparentRim.shader
注意代码中用了Resources.Load,所以必须放在这个目录里,你可以改成别的方式
2 场景中创建一个Sphere(球体),挂上Runner脚本,运行,点击屏幕任意位置,球体就会表现出受击的边缘光效果了

运行效果

代码

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>
  /// 设置材质颜色
  /// </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);
    }
  }
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Unity利用材质自发光实现物体闪烁

    Unity中利用材质自发光实现物体闪烁效果,供大家参考,具体内容如下 补充:这种方法有一点问题,在测试(Windows平台)的时候发现,要想在Build出来的游戏中实现闪烁效果,就必须在 Project 窗口中将源材质的自发光属性(Emission)启用,否则自发光效果就只能在编辑器模式中生效. 启用源材质的自发光效果后,将其亮度(Brightness)调整为0,物体看起来就和没有启用自发光时一样. 看到别的游戏里有物体高亮闪烁效果,但自己不会写Shader,就只想到用材质自发光来做一下,不知道

  • Unity实现UI光晕效果(发光效果)

    Unity中,我们怎么制作UI物体发光的渐隐渐现的效果呢? 比如说我们有一张月亮光晕的精灵图片 我们可以给它添加一个CanvasGroup组件 我们可以发现,组件上的Alpha值可以控制图片的透明度,从0到1 那么我们可以在代码中通过控制Alpha值循环的变化实现发光的光晕效果 以下是代码: using System.Collections; using System.Collections.Generic; using UnityEngine; public class MoonFlash :

  • unity shader实现较完整光照效果

    本文实例为大家分享了unity shader实现光照效果的具体代码,供大家参考,具体内容如下 效果图: shader被附给了球. 灯光需要在属性面板开启阴影. // Upgrade NOTE: replaced 'mul(UNITY_MATRIX_MVP,*)' with 'UnityObjectToClipPos(*)' Shader "Unlit/lightFull" { Properties { _MainTex ("Texture", 2D) = "

  • Unity实现角色受击身体边缘发光特效

    游戏中经常需要制作角色受击打的身体边缘光效果,本文使用的方法是,给Renderer叠加一个制作好的边缘光材质球,并通过脚本动态控制边缘光的渐变效果,表现出受击后的边缘光效果 工程结构如下 1 创建一个材质球HittedMatEffect.mat放在Assets/Resources/Material目录中,使用TransparentRim.shader 注意代码中用了Resources.Load,所以必须放在这个目录里,你可以改成别的方式 2 场景中创建一个Sphere(球体),挂上Runner脚

  • javascript实现鼠标点击生成文字特效

    前端实用脚本(鼠标点击生成文字特效) 你好! 这是你第一次写 博客 目的是为了记录常用到的技术知识以供同行参考学习和方便自己以后查阅 前言 最近在浏览一些博客的时候总是能够看到,当我点击鼠标的时候鼠标位置就会出现文字然后往上方浮动.当时比较好奇,然后空闲时间就去百度了一下这种效果是怎么实现的,然后结合一些博客整理了一下. 旧版本 1.单一的浮动效果 ,旧版本的只有一种颜色,比较单调: 2.没有遵循 : 可以run的demo才是好demo的原则 新版本 1.筛选了多种好看的颜色样式 2.复制代码到

  • jQuery点击出现爱心特效

    本文实例为大家分享了jQuery点击出现爱心特效的具体代码,供大家参考,具体内容如下 代码: <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8" /> <title>爱心效果</title> <script type="text/javascript" src="jquery-3.2.1.

  • Unity实现模型点击事件的方法

    模型点击事件监听 触发模型点击事件的必要条件 需要触发模型点击事件的模型身上必须要挂载Collider 组件 方法一 通过 OnMouseDown 函数监听(只能在PC端有效) 1.在Hierarchy 面板中右键,点击 3D Object->Cube 按钮,创建一个 Cube 模型 2.新建一个脚本,命名为"Test.cs"(代码如下) using System.Collections; using System.Collections.Generic; using Unity

  • 手机端点击图片放大特效PhotoSwipe.js插件实现

    PhotoSwipe插件能实现手机端点击图片全屏放大 再双击图片放大等功能 PhotoSwipe插件官方网站 http://www.photoswipe.com/ 但有一点不太好的是图片放大后再单击不能关闭浏览,要点击关闭按钮或者滑动才能关闭,找了好久配置项都没说到这点上的,只能自己动手改了. 打开photoswipe.js,大概在3179行有个关于tap的函数定义 在开头先定义一个变量 var tap_num = 0; 然后在onTapStart的定义里加入 //根据需求自己添加的S //判断

  • 移动端点击图片放大特效PhotoSwipe.js插件实现

    PhotoSwipe插件能实现手机端点击图片全屏放大 再双击图片放大等功能 PhotoSwipe插件官方网站 http://www.photoswipe.com/ photoswipe之移动端图片放大查看,滑动切换下一张,图片保存到本地. <style> .pnav{margin-top:30px;text-align:center;line-height:24px; font-size:16px} .pnav a{padding:4px} .pnav a.cur{background:#00

  • JS实现页面鼠标点击出现图片特效

    本文实例为大家分享了js实现页面鼠标点击出现图片,供大家参考,具体内容如下 需求: 在页面可视区鼠标点击时,鼠标位置出现图片 技术: 监听器,鼠标坐标获取 效果图 源码分享: 图片是动态添加进页面的,所以没有HTML部分. JS let div = document.createElement("div"); div.id = "mouseImg"; for (let i =0 ; i <3 ;i++){ let img = document.createEl

  • (兼容ff/ie)td点击背景变色特效

    .class1{ background:#efefef; text-align:center; width:80px; } .class2{ background: #ffcc00; text-align:center; width: 80px; } .class3{ background: #ffffff; } .class4{ background: #ffff00; } function addEvent(elm, evType, fn, useCapture){ if (elm.addE

  • Unity iOS混合开发界面切换思路解析

    思路 之前一篇文章里面只谈到了Unity和iOS工程的融合,并没有谈到iOS和Unity界面的切换,这里谈谈思路,Unity导出的iOS工程里面的结构大致是这样的,有一个Window,Window上有一个UnityView,但是并没有控制器,也没有根控制器,虽然在导出的iOS工程中Classes文件夹下的UnityAppController中有rootController的属性,但是上面也标注为空~ 所以,思路就只有一种,,既然Unity导出的iOS工程有一个Window并没有控制器,那好,混合

随机推荐