u3d27-shader-镜面反射
u3d27-shader-镜面反射
说实话镜面反射很简单,难就难在实现镜面反射的代码困难,但是现在我们的代码都是集成好的所以直接使用就可以了!
我们创建一个组件作为我们要变成镜面的工具(一般我们选择plane),之后创建一个材质,在其属性界面
选择我们shader创建的方法
把我们创建的材质赋给我们要做成镜面的组件(plane),把我们的代码与我们的组件绑定!
Ok我们的镜面就做好了!
共享一段JS代码。。。。可以让我们创建的组件自行旋转欧! function Update () {
transform.Rotate( 0, Time.deltaTime * 10, 0 );
}
把代码共享如下:
Shader文件:
Shader "FX/Mirror Reflection"
{Properties
{_MainTex("Base(RGB)",2D)="white"{}
_ReflectionTex("Reflection",2D)="white"{TexGen ObjectLinear}
}
// two texture cards: full thing
Subshader
{Pass
{SetTexture[_MainTex]{combine texture}
SetTexture[_ReflectionTex]{matrix [_ProjMatrix] combine texture*previous} }
}
// fallback: just main texture
Subshader
{Pass
{SetTexture [_MainTex]{combine texture}
}
}
}
Cs文件:
using UnityEngine;
using System.Collections;
//实际上 This is in fact just the Water script from Pro Standard Assets,
// just with refraction stuff removed.
[ExecuteInEditMode]// Make mirror live-update even when not in play mode
public class MirrorReflection:MonoBehaviour
{public bool m_DisablePixelLights=true;
public int m_TextureSize=256;
public float m_ClipPlaneOffset=0.07f;
public LayerMask m_ReflectLayers=-1;
private Hashtable m_ReflectionCameras=new Hashtable(); // Camera ->Camera table private RenderTexture m_ReflectionTexture=null;
private int m_OldReflectionTextureSize=0;
private static bool s_InsideRendering=false;
// This is called when it's known that the object will be rendered by some
// camera. We render reflections and do other updates here.
// Because the script executes in edit mode,reflections for the scene view
// camera will just work!
public void OnWillRenderObject()
{if(!enabled || !renderer || !renderer.sharedMaterial || !renderer.enabled )return;
Camera cam=Camera.current;
if(!cam )return;
// Safeguard from recursive reflections.
if(s_InsideRendering )return;
s_InsideRendering=true;
Camera reflectionCamera;
CreateMirrorObjects(cam,out reflectionCamera );
// find out the reflection plane:position and normal in world space
Vector3 pos=transform.position;
Vector3 normal=transform.up;
// Optionally disable pixel lights for reflection
int oldPixelLightCount=QualitySettings.pixelLightCount;
if(m_DisablePixelLights )QualitySettings.pixelLightCount=0;
UpdateCameraModes(cam,reflectionCamera );
// Render reflection
// Reflect camera around reflection plane
float d=-Vector3.Dot(normal,pos)-m_ClipPlaneOffset;
Vector4 reflectionPlane=new Vector4(normal.x,normal.y,normal.z,d);
Matrix4x4 reflection=Matrix4x4.zero;
CalculateReflectionMatrix(ref reflection,reflectionPlane);
Vector3 oldpos=cam.transform.position;
Vector3 newpos=reflection.MultiplyPoint(oldpos );
reflectionCamera.worldToCameraMatrix=cam.worldToCameraMatrix * reflection;
// Setup oblique projection matrix so that near plane is our reflection
// plane. This way we clip everything below/above it for free.
Vector4 clipPlane=CameraSpacePlane(reflectionCamera,pos,normal,1.0f );
Matrix4x4 projection=cam.projectionMatrix;
CalculateObliqueMatrix(ref projection,clipPlane);
reflectionCamera.projectionMatrix=projection;
reflectionCamera.cullingMask=~(1
reflectionCamera.targetTexture=m_ReflectionTexture;
GL.SetRevertBackfacing(true);
reflectionCamera.transform.position=newpos;
Vector3 euler=cam.transform.eulerAngles;
reflectionCamera.transform.eulerAngles=new Vector3(0,euler.y,euler.z);
reflectionCamera.Render();
reflectionCamera.transform.position=oldpos;
GL.SetRevertBackfacing(false);
Material[]materials=renderer.sharedMaterials;
foreach(Material mat in materials ){
if(mat.HasProperty("_ReflectionTex"))
mat.SetTexture("_ReflectionTex",m_ReflectionTexture );
}
// Set matrix on the shader that transforms UVs from object space into screen
// space. We want to just project reflection texture on screen.
Matrix4x4 scaleOffset=Matrix4x4.TRS(
new Vector3(0.5f,0.5f,0.5f),Quaternion.identity,new Vector3(0.5f,0.5f,0.5f));
Vector3 scale=transform.lossyScale;
Matrix4x4 mtx=transform.localToWorldMatrix * Matrix4x4.Scale(new Vector3(1.0f/scale.x,1.0f/scale.y,1.0f/scale.z));
mtx=scaleOffset * cam.projectionMatrix * cam.worldToCameraMatrix * mtx;
foreach(Material mat in materials ){
mat.SetMatrix("_ProjMatrix",mtx );
}
// Restore pixel light count
if(m_DisablePixelLights )
QualitySettings.pixelLightCount=oldPixelLightCount;
s_InsideRendering=false;
}
// Cleanup all the objects we possibly have created
void OnDisable()
{
if(m_ReflectionTexture ){
DestroyImmediate(m_ReflectionTexture );
m_ReflectionTexture=null;
}
foreach(DictionaryEntry kvp in m_ReflectionCameras )
DestroyImmediate(((Camera)kvp.Value).gameObject );
m_ReflectionCameras.Clear();
}
private void UpdateCameraModes(Camera src,Camera dest )
{
if(dest==null )
return;
// set camera to clear the same way as current camera
dest.clearFlags=src.clearFlags;
dest.backgroundColor=src.backgroundColor;
if(src.clearFlags==CameraClearFlags.Skybox )
{
Skybox sky=src.GetComponent(typeof(Skybox))as Skybox;
Skybox mysky=dest.GetComponent(typeof(Skybox))as Skybox;
if(!sky || !sky.material ){
mysky.enabled=false;
}
else
{
mysky.enabled=true;
mysky.material=sky.material;
}
}
// update other values to match current camera.
// even if we are supplying custom camera&projection matrices,
// some of values are used elsewhere(e.g. skybox uses far plane)
dest.farClipPlane=src.farClipPlane;
dest.nearClipPlane=src.nearClipPlane;
dest.orthographic=src.orthographic;
dest.fieldOfView=src.fieldOfView;
dest.aspect=src.aspect;
dest.orthographicSize=src.orthographicSize;
}
// On-demand create any objects we need
private void CreateMirrorObjects(Camera currentCamera,out Camera reflectionCamera ) {
reflectionCamera=null;
// Reflection render texture
if(!m_ReflectionTexture || m_OldReflectionTextureSize !=m_TextureSize )
{
if(m_ReflectionTexture )
DestroyImmediate(m_ReflectionTexture );
m_ReflectionTexture=new RenderTexture(m_TextureSize,m_TextureSize,16 );
m_ReflectionTexture.name="__MirrorReflection" + GetInstanceID();
m_ReflectionTexture.isPowerOfTwo=true;
m_ReflectionTexture.hideFlags=HideFlags.DontSave;
m_OldReflectionTextureSize=m_TextureSize;
}
// Camera for reflection
reflectionCamera=m_ReflectionCameras[currentCamera]as Camera;
if(!reflectionCamera )// catch both not-in-dictionary and in-dictionary-but-deleted-GO
{
GameObject go=new GameObject("Mirror Refl Camera id" + GetInstanceID()+ " for " + currentCamera.GetInstanceID(),typeof(Camera),typeof(Skybox));
reflectionCamera=go.camera;
reflectionCamera.enabled=false;
reflectionCamera.transform.position=transform.position;
reflectionCamera.transform.rotation=transform.rotation;
reflectionCamera.gameObject.AddComponent("FlareLayer");
go.hideFlags=HideFlags.HideAndDontSave;
m_ReflectionCameras[currentCamera]=reflectionCamera;
}
}
// Extended sign:returns -1,0 or 1 based on sign of a
private static float sgn(float a)
{if(a>0.0f)return 1.0f;
if(a
return 0.0f;
}
// Given position/normal of the plane,calculates plane in camera space.
private Vector4 CameraSpacePlane(Camera cam,Vector3 pos,Vector3 normal,float sideSign)
{
Vector3 offsetPos=pos + normal * m_ClipPlaneOffset;
Matrix4x4 m=cam.worldToCameraMatrix;
Vector3 cpos=m.MultiplyPoint(offsetPos );
Vector3 cnormal=m.MultiplyVector(normal ).normalized * sideSign;
return new Vector4(cnormal.x,cnormal.y,cnormal.z,-Vector3.Dot(cpos,cnormal));
}
//调整给定的射影矩阵以便最近的平面Adjusts the given projection matrix so that near plane is the given clipPlane
// clipPlane is given in camera space. See article in Game Programming Gems 5.
private static void CalculateObliqueMatrix(ref Matrix4x4 projection,Vector4 clipPlane) {Vector4 q=projection.inverse * new Vector4(sgn(clipPlane.x),sgn(clipPlane.y),1.0f,1.0f); Vector4 c=clipPlane *(2.0F /(Vector4.Dot(clipPlane,q)));
// third row=clip plane -fourth row
projection[2]=c.x -projection[3];
projection[6]=c.y -projection[7];
projection[10]=c.z -projection[11];
projection[14]=c.w -projection[15];
}
//围绕给定的平面计算折射矩阵
private static void CalculateReflectionMatrix(ref Matrix4x4 reflectionMat,Vector4 plane) {reflectionMat.m00=(1F -2F*plane[0]*plane[0]);
reflectionMat.m01=(-2F*plane[0]*plane[1]);
reflectionMat.m02=(-2F*plane[0]*plane[2]);
reflectionMat.m03=(-2F*plane[3]*plane[0]);
reflectionMat.m10=(-2F*plane[1]*plane[0]);
reflectionMat.m11=(1F -2F*plane[1]*plane[1]);
reflectionMat.m12=(-2F*plane[1]*plane[2]);
reflectionMat.m13=(-2F*plane[3]*plane[1]);
} reflectionMat.m21=(-2F*plane[2]*plane[1]); reflectionMat.m22=(1F -2F*plane[2]*plane[2]); reflectionMat.m23=(-2F*plane[3]*plane[2]); reflectionMat.m30=0F; reflectionMat.m31=0F; reflectionMat.m32=0F; reflectionMat.m33=1F; }
相关文章
- 5.2光的反射
- 初二物理第四章课上习题答案
- 试卷命题与分析
- 阳极氧化膜性能测试及国家标准
- [光现象]测试题
- 七年级下册科学第二章测试卷
- 八年级物理光的反射练习题
- 上海八年级平面镜成像复习
- [2.2光的反射]导学案
5.2 光的反射 课程目标 1. 认识光反射的规律,了解法线.入射光线.反射光线.入射角.反射角的含义. 2. 理解光的反射定律,理解反射现象中光路的可逆性. 3. 知道镜面反射和漫反射的区别及其应用. 4. 会根据光的反射定律绘图. 知识 ...
初二物理第四章课上练习 1. 小明同学在课外用易拉罐做成如图1所示的装置做小孔成像实验,如果易拉罐 底部有一个很小的三角形小孔,则他在半透明纸上看到的像是:( B ) A .蜡烛的正立像 B .蜡烛的倒立像 C .三角形光斑 D .圆形光斑 ...
八年级物理第二章<光现象>测验题 命题计划与试卷分析 一.考试目的: 1.考查学生在第二章里的学习成就.教师完成教学目标的程度: 2.检查学生在一段时间内(第二章光现象的学习期间)对所学知识的掌握程度: 3.激发学生的学习行为. ...
阳极氧化膜性能测试方法 1. 光泽 1.1 目视法 目视检测法:包含对颜色.色差.表面光泽和表面表面缺陷的检测.其观察距离一般是0.5m:(GB/T14952.3-1994) 1.2 光泽仪 由于光泽目视时无法量化,所以采用了相应的仪器:光 ...
光现象单元测试题 班级_________ 姓名_________ 学号_________ 成绩__________ 一 .填空题(每空1分,共27分) 1.影子和平面镜成像的原理不同,影子是光的 形成的,平面镜成像是光的 形成的. 2.光的 ...
七年级下科学第二章测试卷 一.选择题 1.著名作家海伦·凯勒失去了听觉和视觉,但她通过"阅读"盲文学习了大量的知识,一生共写了<我的生活>等14部著作.她获取信息的主要手段是 A.触觉 B.味觉 C.嗅觉 D ...
光的反射 一.选择题 (1)关于光的反射,下列说法错误的是( ) A .反射光线跟入射光线和法线在同一平面内B .反射光线和入射光线分居在法线的两侧C .入射角增大,反射角也同样增大D .入射角大于反射角 (2)入射光线与反射光线间的夹角为 ...
平面镜成像 1.在"探究平面镜成像特点"的实验中,下列说法正确的是( ) A.实验最好在较暗的环境中进行 B.把光屏放在玻璃板后面像的位置,光屏上有像出现 C.将蜡烛向玻璃板靠近时像会变大 D.物体靠近玻璃板时,所成的像 ...
<2.2 光的反射>导学案 编写者:刘勇 审批者: 时间:2010-9-21 一.学习目标: 1.通过一些生活现象知道物体的表面可以发生反射现象. 2.通过实验得出光的反射定律,知道光路是可逆的,并能解释有关问题. 2.知道光的 ...