1 原理

凸镜贴图渐变凸镜贴图 中介绍了使用 OpenGL 实现凸镜贴图及其原理,通过顶点坐标映射到纹理坐标,并构造三角形网格,构建了真正的三维凸镜模型。本文通过 Shader 实现半球卷屏特效,通过屏幕坐标映射到纹理坐标,不需要构建凸镜模型,效率更高。

1)凸变换原理

​ 以下凸变换的原理图及公式推导,该图是截面图,vertex 是屏幕坐标,texture 是纹理坐标。

​ 注意:屏幕坐标原坐标原点在屏幕左上角,y 轴向下,x、y 轴的值域分别为 [0, ScreenWidth]、[0, ScreenHeight],纹理坐标原坐标原点在纹理图片左下角,x、y 轴的值域都是 [0, 1],这里已进行了一些预处理,将屏幕坐标和纹理坐标的坐标轴都变换到中心位置,x、y 轴的值域都变换到 [-1, 1]。

2)渐变原理

​ 当凸镜半角 (φ/2) 较小时(近似0°),凸镜半径较大 (近似无穷大),屏幕只需要贴到凸镜的很小一块区域,该区域近似一个平面;当凸镜半角 (φ/2) 较大时(等于90°),凸镜半径较大(等于 1/sin(φ/2)),屏幕贴满整个凸镜;当凸镜半角 (φ/2) 由 0° 渐变到 90° 时,就会看到屏幕逐渐卷曲的效果。

​ 本文代码资源见→Unity3D半球卷屏特效

2 代码实现

​ CurlEffect.cs

using UnityEngine;

[RequireComponent(typeof(Camera))]  // 屏幕后处理特效一般都需要绑定在像机上
public class CurlEffect : MonoBehaviour {
public float curlSpeed = 0.4f; // 卷屏速度
private Material smallConvexMaterial; // 小凸变换材质
private Material largeConvexMaterial; // 大凸变换材质
private bool enableSmallConvex = false; // 小凸变换开关
private bool enableLargeConcave = false; // 大凸变换开关
private float alpha; // 卷屏凸镜渐变半角 private void Awake() {
smallConvexMaterial = new Material(Shader.Find("Custom/Curl/SmallConvex"));
largeConvexMaterial = new Material(Shader.Find("Custom/Curl/LargeConvex"));
smallConvexMaterial.hideFlags = HideFlags.DontSave;
largeConvexMaterial.hideFlags = HideFlags.DontSave;
} private void Update() {
if (Input.GetMouseButton(0)) {
alpha = 0.01f;
curlSpeed = Mathf.Abs(curlSpeed);
enableSmallConvex = true;
enableLargeConcave = false;
}
} private void OnRenderImage (RenderTexture source, RenderTexture destination) {
if (enableSmallConvex) {
smallConvexMaterial.SetFloat("_alpha", alpha);
IncreaseAlpha();
Graphics.Blit (source, destination, smallConvexMaterial);
} else if (enableLargeConcave) {
largeConvexMaterial.SetFloat("_alpha", alpha);
IncreaseAlpha();
Graphics.Blit (source, destination, largeConvexMaterial);
} else {
Graphics.Blit (source, destination);
}
} private void IncreaseAlpha() { // alpha自增
alpha += Time.deltaTime * curlSpeed;
if (alpha > Mathf.PI / 2 && curlSpeed > 0) {
alpha = Mathf.PI / 2;
curlSpeed = -curlSpeed;
} else if (alpha < 0.01f && curlSpeed < 0) {
alpha = 0.01f;
curlSpeed = -curlSpeed;
enableSmallConvex = !enableSmallConvex; // 大凸镜和小凸镜交替执行
enableLargeConcave = ! enableLargeConcave;
}
}
}

​ SmallConvex.shader

Shader "Custom/Curl/SmallConvex" // 小凸镜变换
{
Properties
{
_MainTex ("mainTex", 2D) = "white" {}
} SubShader
{
Pass
{
ZTest Always
Cull Off
ZWrite Off
Fog { Mode off } CGPROGRAM #pragma vertex vert_img // UnityCG.cginc中定义了vert_img方法, 对vertex和texcoord进行了处理, 输出v2f_img中的pos和uv
#pragma fragment frag
#pragma fragmentoption ARB_precision_hint_fastest #include "UnityCG.cginc" sampler2D _MainTex;
float _alpha; float2 beforeConvex(float2 pos)
{ // 小凸化前置变换, 将pos的窄边映射到(-1, 1)之间
pos /= _ScreenParams.xy; // 坐标映射到(0, 1)之间
pos = pos * 2 - 1; // 坐标映射到(-1, 1)之间
pos.y = -pos.y; // 屏幕坐标系原点在左上角, y轴向下, 所以要取反
pos.x *= (_ScreenParams.x / _ScreenParams.y); // 窄边映射到(-1, 1)之间, 宽边映射到(-ratio, ratio)之间(ratio为屏幕宽高比)
return pos;
} float2 convex(float2 pos)
{ // 凸化变换, 将屏幕坐标映射到纹理坐标, 窄边映射到(-1, 1)之间, 宽边大致映射到(-ratio, ratio)之间(ratio为屏幕宽高比)
float rho = length(pos);
float beta = rho * sin(_alpha);
if (beta > 1)
{
return float2(-10000, -1000000);
}
return pos * asin(beta) / _alpha / rho;
} float2 afterConvex(float2 uv)
{ // 小凸化后置变换, 将uv的窄边和宽边都映射到(0, 1)之间
uv.x = uv.x / (_ScreenParams.x / _ScreenParams.y) / 2 + 0.5; // 坐标由(-ratio, ratio)还原到(0, 1)
uv.y = uv.y / 2 + 0.5; // 坐标由(-1, 1)还原到(0, 1)
return uv;
} fixed4 frag(v2f_img i) : SV_Target // uv坐标的计算不能在顶点着色器中进行, 因为屏后处理的顶点只有屏幕的4个角顶点
{
float2 pos = beforeConvex(i.pos.xy);
float2 uv = convex(pos);
uv = afterConvex(uv);
if (uv.x < 0 || uv.y < 0 || uv.x > 1 || uv.y > 1)
{
return fixed4(0, 0, 0, 0);
}
return tex2D(_MainTex, uv);
} ENDCG
}
} Fallback off
}

​ LargeConvex.shader

Shader "Custom/Curl/LargeConvex" // 大凸镜变换
{
Properties
{
_MainTex ("mainTex", 2D) = "white" {}
} SubShader
{
Pass
{
ZTest Always
Cull Off
ZWrite Off
Fog { Mode off } CGPROGRAM #pragma vertex vert_img // UnityCG.cginc中定义了vert_img方法, 对vertex和texcoord进行了处理, 输出v2f_img中的pos和uv
#pragma fragment frag
#pragma fragmentoption ARB_precision_hint_fastest #include "UnityCG.cginc" sampler2D _MainTex;
float _alpha; float2 beforeConvex(float2 pos)
{ // 大凸化前置变换, 将pos的宽边映射到(-1, 1)之间
pos /= _ScreenParams.xy; // 坐标映射到(0, 1)之间
pos = pos * 2 - 1; // 坐标映射到(-1, 1)之间
pos.y = -pos.y; // 屏幕坐标系原点在左上角, y轴向下, 所以要取反
pos.y /= (_ScreenParams.x / _ScreenParams.y); // 宽边映射到(-1, 1)之间, 窄边映射到(-1/ratio, 1/ratio)之间(ratio为屏幕宽高比)
return pos;
} float2 convex(float2 pos)
{ // 凸化变换, 将屏幕坐标映射到纹理坐标, 宽边映射到(-1, 1)之间, 窄边大致映射到(-1/ratio, 1/ratio)之间(ratio为屏幕宽高比)
float rho = length(pos);
float beta = rho * sin(_alpha);
if (beta > 1)
{
return float2(-10000, -1000000);
}
return pos * asin(beta) / _alpha / rho;
} float2 afterConvex(float2 uv)
{ // 大凸化后置变换, 将uv的宽边和窄边都映射到(0, 1)之间
uv.x = uv.x / 2 + 0.5; // 坐标由(-1, 1)还原到(0, 1)
uv.y = uv.y * (_ScreenParams.x / _ScreenParams.y) / 2 + 0.5; // 坐标由(-ratio, ratio)还原到(0, 1)
return uv;
} fixed4 frag(v2f_img i) : SV_Target // uv坐标的计算不能在顶点着色器中进行, 因为屏后处理的顶点只有屏幕的4个角顶点
{
float2 pos = beforeConvex(i.pos.xy);
float2 uv = convex(pos);
uv = afterConvex(uv);
if (uv.x < 0 || uv.y < 0 || uv.x > 1 || uv.y > 1)
{
return fixed4(0, 0, 0, 0);
}
return tex2D(_MainTex, uv);
} ENDCG
}
} Fallback off
}

3 运行效果

4 推荐阅读

​ 声明:本文转自【Unity3D】半球卷屏特效

【Unity3D】半球卷屏特效的更多相关文章

  1. unity3d 游戏插件 溶解特效插件 - Dissolve Shader

    unity3d 游戏插件 溶解特效插件 - Dissolve Shader   链接: https://pan.baidu.com/s/1hr7w39U 密码: 3ed2

  2. 【SecureCRT配置】修改默认卷屏行数当做一个操作,屏幕输出有上百行,当需要将屏幕回翻时,这个设置会有很大帮助,默认为500行,可以改为10000行,不用担心找不到了。 选项 => 全局选项 => Default Session => Edit Default Settings => Terminal => Emulation => Scrollback 修改为32000。

    SecureCRT配置屏幕内容输出到log文件 SecureCRT看不到前几分钟操作的内容,或者想把通过vi命令查看的日志输出到log文件(在懒得下载日志文件的情况下),所以接下来就这样操作: 文件保 ...

  3. Unity3D去掉全屏时的屏幕黑边

    给全屏后不在乎拉伸变形仍想让画面占满屏幕的朋友,网上搜了一上午,实在是没有相关的资料,只能自己琢磨了. 使用Canvas Scaler在全屏后Unity虽然会为我们自动拉伸UI,但拉伸后仍然保持我们在 ...

  4. unity3d笔记:控制特效的播放速度

           一般在游戏中,主角或者怪物会受到减速效果,或者攻击速度减慢等类似的状态.本身动作减速的同时,衔接在角色上的特效也需要改变相应的播放速度.一般特效有三个游戏组件:   关键点就是改变Ani ...

  5. Unity3D在Windows的全屏和跨屏(双屏)方案

    方案1 unity中2个摄像机场景显示在两个显示器屏幕上(一个窗口跨屏) 1.设置场景中的两个摄像机 摄像机1 摄像机2 2.设置发布的平台及分辨率 3.全屏运行游戏,没有标题栏还可以通过-popup ...

  6. Jquery实现手机上下滑屏滑动的特效代码

    要引入两个jquery插件 可以去网上下载 <script src="jquery-1.11.1.min.js"></script><script s ...

  7. 全屏显示网页FULLSCREEN API

    第一次看到应用 Fullscreen API 全屏显示网页,是 FaceBook 中的照片放大.作为一个比较新的 API,目前只有 Safari.Chrome 和 FireFox 三种浏览器支持该特性 ...

  8. 用Qt(C++)实现如苹果般的亮屏效果

    用Qt(C++)实现如苹果般的亮屏效果 苹果的亮屏效果可能有很多人没注意到,和其他大部分手机或电脑不同的是,苹果的亮屏特效不是简单的亮度变化,而是一个渐亮的过程.详细来说就是,图片中较亮的部分先显示出 ...

  9. GNU Emacs命令速查表

    GNU Emacs命令速查表 第一章  Emacs的基本概念 表1-1:Emacs编辑器的主模式 模式 功能 基本模式(fundamental mode) 默认模式,无特殊行为 文本模式(text m ...

  10. Unity中的ShaderToys——将大神们写的shader搬到unity中来吧

    http://lib.csdn.net/article/unity3d/38699 这篇文章翻译自国外的一篇文章(这里是原文链接),正在使用unity的你是否在shader toy上发现很多牛逼哄哄的 ...

随机推荐

  1. WebApi使用Swagger

    services.AddSwaggerGen(c => { c.SwaggerDoc("v1", new OpenApiInfo { Title = "My API ...

  2. [转帖]TiDB 数据库核心原理与架构 [TiDB v6](101)笔记

    https://www.jianshu.com/p/01e49a93f671 description: "本课程专为将在工作中使用 TiDB 数据库的开发人员.DBA 和架构师设计. 本门课 ...

  3. [转帖]Sosreport:收集系统日志和诊断信息的工具

    https://zhuanlan.zhihu.com/p/39259107 如果你是 RHEL 管理员,你可能肯定听说过 Sosreport :一个可扩展.可移植的支持数据收集工具.它是一个从类 Un ...

  4. [转帖]jar启动指定JDK/JRE 安装路径教程

    https://blog.csdn.net/weixin_40986713/article/details/128136777 前言 因为疫情在家办公的缘故,有个老项目,需要改个接口,然后需要前端联调 ...

  5. 不同信创服务器Redis7.0.5性能表现总结

    不同信创服务器Redis7.0.5性能表现总结 背景以及基础约定 随着美帝2022.10收紧EAR规定的硬件出口规定 信创事业迎来了一波新的高潮. 最近不仅仅要求国产化的硬件. 更要求国产化的OS,以 ...

  6. 使用rpmbuild打包erlang和rabbitmq进行部署服务的方法

    使用rpmbuild打包erlang和rabbitmq进行部署服务的方法 背景说明 1. rabbitmq 是基于 erlang 开发的消息列队, 本身rabbitmq 自己不区分架构. 2. 但是e ...

  7. js文件下载blob

    使用axios文件下载 if (tableDataSource.selectedRowKeys.length > 0) { //本次请求你携带token axios.defaults.heade ...

  8. gpedit.msc 打不开

    win10系统推出已有不短的时间了,朋友们也纷纷升级了win10系统,但是暴露的问题也是越来越多,比如win10系统打开运行输入gpedit.msc命令时却提示找不到文件.那出现win10打不开gpe ...

  9. Qt "有效且启用的储存库"问题

    传送门 : https://www.cnblogs.com/SaveDictator/p/8532664.html 看就完了, 反正我好了 https://mirrors.tuna.tsinghua. ...

  10. 使用CSS3实现鼠标移到图片上图片放大

    转自 http://www.webkaka.com/tutorial/html/2017/072731/ 在现在的网页设计中,鼠标移到图片上图片放大的效果常常被用到,这个效果多应用于文章列表里.我一开 ...