边缘检测(一)
作者:追风剑情 发布于:2016-11-19 0:45 分类:Shader
一、创建脚本(需挂在摄像机上)
using UnityEngine;
using System.Collections;
/// <summary>
/// 边缘检测
/// 把该脚本拖曳到摄像机上。
/// 把边缘检测的Shader拖到edgeDetectShader上。
/// </summary>
public class EdgeDetection : PostEffectsBase {
//调整描边强度
[Range(0.0f, 1.0f)]
public float edgesOnly = 0.0f;
//描边颜色
public Color edgeColor = Color.black;
//背景颜色
public Color backgroundColor = Color.white;
public Shader edgeDetectShader;
private Material edgeDetectMaterial = null;
public Material material {
get {
edgeDetectMaterial = CheckShaderAndCreateMaterial(edgeDetectShader, edgeDetectMaterial);
return edgeDetectMaterial;
}
}
void OnRenderImage(RenderTexture src, RenderTexture dest) {
if(material != null){
material.SetFloat("_EdgeOnly", edgesOnly);
material.SetColor("_EdgeColor", edgeColor);
material.SetColor("_BackgroundColor", backgroundColor);
Graphics.Blit(src, dest, material);
}else{
Graphics.Blit(src, dest);
}
}
}
二、创建Shader
//屏幕后处理效果——边缘检测(实现描边效果)
//需要注意的是,本Shader实现的边缘检测仅仅利用了屏幕颜色信息,
//而在实际应用中,物体的纹理、阴影等信息均会影响边缘检测的结果,
//使得结果包含许多非预期的描边。为了得到更加准确的边缘信息,我们往往
//会在屏幕的深度纹理和法线纹理上进行边缘检测。
Shader "Custom/Chapter12-EdgeDetection" {
Properties {
_MainTex ("Base (RGB)", 2D) = "white" {}
//边缘线强度
_EdgeOnly ("Edge Only", float) = 1.0
//边缘颜色
_EdgeColor ("Edge Color", Color) = (0, 0, 0, 1)
//背景颜色
_BackgroundColor ("Background Color", Color) = (1, 1, 1, 1)
//说明:
//当_EdgeOnly值为0时,边缘将会叠加在原渲染图像上;
//当_EdgeOnly值为1时,则会只显示边缘,不显示原渲染图像。
//其中,背景颜色由_BackgroundColor指定,边缘颜色由_EdgeColor指定。
}
SubShader {
Pass {
Tags { "LightMode"="ForwardBase" }
ZTest Always Cull Off ZWrite Off
CGPROGRAM
#pragma vertex vert
#pragma fragment fragSobel
#include "UnityCG.cginc"
sampler2D _MainTex;
//xxx_TexelSize是Unity为我们提供的访问xxx纹理对应的每个纹素的大小。
//例如,一张512x512大小的纹理,该值大约为0.001953(即1/512)。
//由于卷积需要对相领区域内的纹理进行采样,因此我们需要利用_MainTex_TexelSize来计算各个
//相邻区域的纹理坐标。
half4 _MainTex_TexelSize;
fixed _EdgeOnly;
fixed4 _EdgeColor;
fixed4 _BackgroundColor;
struct v2f {
float4 pos : SV_POSITION;
half4 uv[5] : TEXCOORD0;
};
v2f vert(appdata_img v) {
v2f o;
UNITY_INITIALIZE_OUTPUT(v2f, o);//初始化参数
o.pos = mul(UNITY_MATRIX_MVP, v.vertex);
half2 uv = v.texcoord;
//我们在v2f结构体中定义了一个维数为9的纹理数组,对应了使用Sobel算子采样时需要的9
//个领域纹理坐标。通过把计算采样纹理坐标的代码从片元着色器中转移到顶点着色器中,可以减少
//运算,提高性能。由于从顶点着色器到片元着色器的插值是线性的,因此这样的转移并不会影响
//纹理坐标的计算结果。
o.uv[0].xy = uv + _MainTex_TexelSize.xy * half2(-1, -1);
o.uv[0].zw = uv + _MainTex_TexelSize.xy * half2( 0, -1);
o.uv[1].xy = uv + _MainTex_TexelSize.xy * half2( 1, -1);
o.uv[1].zw = uv + _MainTex_TexelSize.xy * half2(-1, 0);
o.uv[2].xy = uv + _MainTex_TexelSize.xy * half2( 0, 0);
o.uv[2].zw = uv + _MainTex_TexelSize.xy * half2( 1, 0);
o.uv[3].xy = uv + _MainTex_TexelSize.xy * half2(-1, 1);
o.uv[3].zw = uv + _MainTex_TexelSize.xy * half2( 0, 1);
o.uv[4].xy = uv + _MainTex_TexelSize.xy * half2( 1, 1);
return o;
}
//计算亮度值
fixed luminance(fixed4 color) {
return 0.2125 * color.r + 0.7154 * color.g + 0.0721 * color.b;
}
//计算当前像素的梯度值
//Sobel函数将利用Sobel算子对原图进行边缘检测。
half Sobel(v2f i) {
//定义水平方向使用的卷积核Gx
const half Gx[9] = {-1, -2, -1,
0, 0, 0,
1, 2, 1};
//定义竖直方向使用的卷积核Gy
const half Gy[9] = {-1, 0, 1,
-2, 0, 2,
-1, 0, 1};
half texColor;
half edgeX = 0;
half edgeY = 0;
int n = 0;
//依次对9个像素进行采样,计算它们的亮度值,再与卷积核Gx和Gy中对应的权重相乘后,
//叠加到各自的梯度值上。
for (uint it = 0; it < 9; it++) {
if(it%2 == 0){
texColor = luminance(tex2D(_MainTex, i.uv[n].xy));
}else{
texColor = luminance(tex2D(_MainTex, i.uv[n].zw));
n++;
}
edgeX += texColor * Gx[it];
edgeY += texColor * Gy[it];
}
//最后,我们从1中减去水平方向和竖直方向的梯度值的绝对值,得到edge。
//edge值越小,表明该位置越可能是一个边缘点。至此,边缘检测过程结束。
half edge = 1 - abs(edgeX) - abs(edgeY);
return edge;
}
//Unity5.x中返回的语义用SV_Target
//使用Sobel算子进行边缘检测,实现搭边效果。
fixed4 fragSobel(v2f i) : COLOR {
//调用Sobel函数计算当前像素的梯度值
half edge = Sobel(i);
//计算背景为原图下的颜色值
fixed4 withEdgeColor = lerp(_EdgeColor, tex2D(_MainTex, i.uv[2].xy), edge);
//计算背景为纯色下的颜色值
fixed4 onlyEdgeColor = lerp(_EdgeColor, _BackgroundColor, edge);
//利用_EdgeOnly在两者之间插值得到最终的像素值。
return lerp(withEdgeColor, onlyEdgeColor, _EdgeOnly);
}
ENDCG
}
}
FallBack Off
}
运行效果
edgesOnly = 0
edgesOnly = 0.5
edgesOnly = 1
标签: Shader
日历
最新文章
随机文章
热门文章
分类
存档
- 2025年11月(1)
- 2025年9月(3)
- 2025年7月(4)
- 2025年6月(5)
- 2025年5月(1)
- 2025年4月(5)
- 2025年3月(4)
- 2025年2月(3)
- 2025年1月(1)
- 2024年12月(5)
- 2024年11月(5)
- 2024年10月(5)
- 2024年9月(3)
- 2024年8月(3)
- 2024年7月(11)
- 2024年6月(3)
- 2024年5月(9)
- 2024年4月(10)
- 2024年3月(11)
- 2024年2月(24)
- 2024年1月(12)
- 2023年12月(3)
- 2023年11月(9)
- 2023年10月(7)
- 2023年9月(2)
- 2023年8月(7)
- 2023年7月(9)
- 2023年6月(6)
- 2023年5月(7)
- 2023年4月(11)
- 2023年3月(6)
- 2023年2月(11)
- 2023年1月(8)
- 2022年12月(2)
- 2022年11月(4)
- 2022年10月(10)
- 2022年9月(2)
- 2022年8月(13)
- 2022年7月(7)
- 2022年6月(11)
- 2022年5月(18)
- 2022年4月(29)
- 2022年3月(5)
- 2022年2月(6)
- 2022年1月(8)
- 2021年12月(5)
- 2021年11月(3)
- 2021年10月(4)
- 2021年9月(9)
- 2021年8月(14)
- 2021年7月(8)
- 2021年6月(5)
- 2021年5月(2)
- 2021年4月(3)
- 2021年3月(7)
- 2021年2月(2)
- 2021年1月(8)
- 2020年12月(7)
- 2020年11月(2)
- 2020年10月(6)
- 2020年9月(9)
- 2020年8月(10)
- 2020年7月(9)
- 2020年6月(18)
- 2020年5月(4)
- 2020年4月(25)
- 2020年3月(38)
- 2020年1月(21)
- 2019年12月(13)
- 2019年11月(29)
- 2019年10月(44)
- 2019年9月(17)
- 2019年8月(18)
- 2019年7月(25)
- 2019年6月(25)
- 2019年5月(17)
- 2019年4月(10)
- 2019年3月(36)
- 2019年2月(35)
- 2019年1月(28)
- 2018年12月(30)
- 2018年11月(22)
- 2018年10月(4)
- 2018年9月(7)
- 2018年8月(13)
- 2018年7月(13)
- 2018年6月(6)
- 2018年5月(5)
- 2018年4月(13)
- 2018年3月(5)
- 2018年2月(3)
- 2018年1月(8)
- 2017年12月(35)
- 2017年11月(17)
- 2017年10月(16)
- 2017年9月(17)
- 2017年8月(20)
- 2017年7月(34)
- 2017年6月(17)
- 2017年5月(15)
- 2017年4月(32)
- 2017年3月(8)
- 2017年2月(2)
- 2017年1月(5)
- 2016年12月(14)
- 2016年11月(26)
- 2016年10月(12)
- 2016年9月(25)
- 2016年8月(32)
- 2016年7月(14)
- 2016年6月(21)
- 2016年5月(17)
- 2016年4月(13)
- 2016年3月(8)
- 2016年2月(8)
- 2016年1月(18)
- 2015年12月(13)
- 2015年11月(15)
- 2015年10月(12)
- 2015年9月(18)
- 2015年8月(21)
- 2015年7月(35)
- 2015年6月(13)
- 2015年5月(9)
- 2015年4月(4)
- 2015年3月(5)
- 2015年2月(4)
- 2015年1月(13)
- 2014年12月(7)
- 2014年11月(5)
- 2014年10月(4)
- 2014年9月(8)
- 2014年8月(16)
- 2014年7月(26)
- 2014年6月(22)
- 2014年5月(28)
- 2014年4月(15)
友情链接
- Unity官网
- Unity圣典
- Unity在线手册
- Unity中文手册(圣典)
- Unity官方中文论坛
- Unity游戏蛮牛用户文档
- Unity下载存档
- Unity引擎源码下载
- Unity服务
- Unity Ads
- wiki.unity3d
- Visual Studio Code官网
- SenseAR开发文档
- MSDN
- C# 参考
- C# 编程指南
- .NET Framework类库
- .NET 文档
- .NET 开发
- WPF官方文档
- uLua
- xLua
- SharpZipLib
- Protobuf-net
- Protobuf.js
- OpenSSL
- OPEN CASCADE
- JSON
- MessagePack
- C在线工具
- 游戏蛮牛
- GreenVPN
- 聚合数据
- 热云
- 融云
- 腾讯云
- 腾讯开放平台
- 腾讯游戏服务
- 腾讯游戏开发者平台
- 腾讯课堂
- 微信开放平台
- 腾讯实时音视频
- 腾讯即时通信IM
- 微信公众平台技术文档
- 白鹭引擎官网
- 白鹭引擎开放平台
- 白鹭引擎开发文档
- FairyGUI编辑器
- PureMVC-TypeScript
- 讯飞开放平台
- 亲加通讯云
- Cygwin
- Mono开发者联盟
- Scut游戏服务器引擎
- KBEngine游戏服务器引擎
- Photon游戏服务器引擎
- 码云
- SharpSvn
- 腾讯bugly
- 4399原创平台
- 开源中国
- Firebase
- Firebase-Admob-Unity
- google-services-unity
- Firebase SDK for Unity
- Google-Firebase-SDK
- AppsFlyer SDK
- android-repository
- CQASO
- Facebook开发者平台
- gradle下载
- GradleBuildTool下载
- Android Developers
- Google中国开发者
- AndroidDevTools
- Android社区
- Android开发工具
- Google Play Games Services
- Google商店
- Google APIs for Android
- 金钱豹VPN
- TouchSense SDK
- MakeHuman
- Online RSA Key Converter
- Windows UWP应用
- Visual Studio For Unity
- Open CASCADE Technology
- 慕课网
- 阿里云服务器ECS
- 在线免费文字转语音系统
- AI Studio
- 网云穿
- 百度网盘开放平台
- 迅捷画图
- 菜鸟工具
- [CSDN] 程序员研修院
- 华为人脸识别
- 百度AR导航导览SDK
- 海康威视官网
- 海康开放平台
- 海康SDK下载
- git download
- Open CASCADE
- CascadeStudio
交流QQ群
-
Flash游戏设计: 86184192
Unity游戏设计: 171855449
游戏设计订阅号











