【Unity3D】卷轴特效

news2025/1/19 16:27:12

1 原理

        当一个圆在地面上沿直线匀速滚动时,圆上固定点的运动轨迹称为旋轮线(或摆线、圆滚线)。本文实现的卷轴特效使用了旋轮线相关理论。

        以下是卷轴特效原理及公式推导,将屏幕坐标 (x) 映射到纹理坐标 (u)。

         注意:屏幕坐标 x 值域为 [0, ScreenWidth],这里已归一化到 [0, 1]。

        本文代码资源见→Unity3D Shader卷轴滚动特效。

2 代码实现

        RollEffect.cs

using UnityEngine;
 
[RequireComponent(typeof(Camera))]  // 屏幕后处理特效一般都需要绑定在像机上
public class RollEffect : MonoBehaviour {
    public float radius = 0.05f; // 圆半径
    public float rollSpeed = 0.8f; // 圆滚动角速度
    private Texture rollTex; // 滚动轴纹理
    private Texture backTex; // 底部背景纹理
    private float rollTime = 0; // 滚动时间
    private float maxRollTime; // 最长滚动时间
    private float rollDirection = 1; // 滚动方向(1: 向右, -1: 向左)
    private Material rollMaterial; // 滚动特效材质
    private bool enableRoll = false; // 滚动特效开关
 
    private void Awake() {
        rollMaterial = new Material(Shader.Find("Custom/Curl/Roll"));
        rollMaterial.hideFlags = HideFlags.DontSave;
        rollTex = Resources.Load<Texture>("RollTex");
        backTex = Resources.Load<Texture>("BackTex");
    }
 
    private void Update() {
        if (Input.GetMouseButton(0)) {
            rollTime = 0;
            maxRollTime = 1 / rollSpeed / radius;
            enableRoll = true;
        }
    }
 
    private void OnRenderImage (RenderTexture source, RenderTexture destination) {
        if (enableRoll) {
            rollMaterial.SetTexture("_RollTex", rollTex);
            rollMaterial.SetTexture("_BackTex", backTex);
            rollMaterial.SetFloat("_theta", rollSpeed);
            rollMaterial.SetFloat("_r", radius);
            rollMaterial.SetFloat("_t", rollTime);
            IncreaseTime();
            Graphics.Blit (source, destination, rollMaterial);
        }  else {
            Graphics.Blit (source, destination);
        }
	}
 
    private void IncreaseTime() { // 时间自增
        rollTime += rollDirection * Time.deltaTime;
        if (rollTime > maxRollTime) {
            rollTime = maxRollTime;
            rollDirection = -rollDirection; // 反向卷轴
        } else if (rollTime < 0) {
            rollTime = 0;
            rollDirection = -rollDirection;
        }
    }
}

        说明: RollEffect 脚本组件需要挂在相机上。

        Roll.shader

Shader "Custom/Curl/Roll" // 小凸镜变换
{
	Properties 
	{
		_MainTex ("mainTex", 2D) = "white" {}
        _RollTex ("rollTex", 2D) = "white" {}
        _BackTex ("backTex", 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;
            sampler2D _RollTex; // 滚动轴纹理
            sampler2D _BackTex; // 底部背景纹理
            float _theta; // 圆滚动角速度
            float _r; // 圆半径
            float _t; // 滚动时间

            float4 roll(float rho, float v)
            { // 滚动变换, 将屏幕坐标映射到纹理坐标
                if (rho < _theta * _r * _t - _r)
                {
                    return tex2D(_BackTex, float2(rho, v));
                }
                else if (rho < _theta * _r * _t)
                {
                    float a = _theta * _r * _t - rho;
                    float phi = acos(a / _r);
                    float u = _theta * _r * _t - (UNITY_HALF_PI + phi) * _r;
                    if (u > 0)
                    {
                        return tex2D(_RollTex, float2(u, v)) * pow(sin(phi), 2);
                    }
                    u = _theta * _r * _t - (UNITY_HALF_PI - phi) * _r;
                    return tex2D(_MainTex, float2(u, v)); // 刚开始卷动时会触发
                }
                else if (rho < _theta * _r * _t + _r)
                {
                    float a = rho - _theta * _r * _t;
                    float phi = acos(a / _r);
                    float u = _theta * _r * _t - (3 * UNITY_HALF_PI - phi) * _r;
                    if (u > 0)
                    {
                        return tex2D(_RollTex, float2(u, v)) * pow(sin(phi), 2);
                    }
                    return tex2D(_MainTex, float2(rho, v)); // 刚开始卷动时会触发
                }
                else
                {
                    return tex2D(_MainTex, float2(rho, v));
                }
            }
 
            fixed4 frag(v2f_img i) : SV_Target // uv坐标的计算不能在顶点着色器中进行, 因为屏后处理的顶点只有屏幕的4个角顶点
            {
                return roll(i.pos.x / _ScreenParams.x, i.uv.y);
            }
 
			ENDCG
		}
	}
 
	Fallback off
}

3 运行效果

4 推荐阅读

  • 渲染管线
  • 固定管线着色器一
  • 固定管线着色器二
  • 表面着色器
  • 顶点和片段着色器
  • 选中物体描边特效
  • 水波特效
  • 半球卷屏特效

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/41194.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Controller Area Network(CAN)简介

文章目录前言-什么是通讯&#xff1f;一、CAN是什么&#xff1f;二、CAN的应用示例CAN网络细分三、CAN发展历史四、汽车网络汇总总结前言-什么是通讯&#xff1f; “通讯是两个或两个以上参与者之间交换信息的有意识活动&#xff0c;目的是通过符号和语义规范的共享系统来发送…

微服务系列之初探“微服务架构”

随笔 有时你必须对你想得到的东西充满敬畏。 参考书籍&#xff1a; “凤凰架构”“微服务架构设计模式” 在了解微服务架构之前我们有必要解答“什么是架构”、“什么是架构的风格”这两个问题&#xff0c;同时需要带着“架构并不是被发明出来的&#xff0c;而是持续演进的…

[AI] LRTA*ls(k)搜索算法

LRTA*LS[K]一、LRTA*(K)算法的缺点二、LRTA∗LS(k)LRTA*_{LS}(k)LRTA∗LS​(k)算法1、选择局部空间2、更新局部空间论文在这里&#xff01; 一、LRTA*(K)算法的缺点 LRTA*(K)算法每次要更新队列Q里的state&#xff0c;但有三点缺陷&#xff1a; 如果state y进入 Q&#xff0c…

【网络工程】6、防火墙介绍及配置实操

接上篇《5、路由器介绍及配置实操》 之前我们讲解了网络设备路由器的介绍&#xff0c;以及完成了路由器的相关配置实操。本篇我们来讲解防火墙的基础知识以及相应的实操案例。 一、什么是防火墙&#xff1f; 防火墙是一个安全产品&#xff0c;它可以把安全的内网和不安全的外网…

【前端】jQuery-概述+基本使用+常用API

目录 一、jQuery概述 1.1JavaScript库 1.2jQuery的概念 二、jQuery的基本使用 2.1jQuery的下载 2.2jQuery的使用步骤 2.3jQuery的入口函数 2.4jQuery的顶级对象$ 2.5jQuery对象和DOM对象 三、jQuery常用API 3.1jQuery选择器 3.1.1jQuery基础选择器 3.1.2jQuery层级选…

Linux下C语言实现HTTP文件服务器和TCP协议实现网络数据传输

在实际开发中经常用到web框架&#xff0c;比如Servlet&#xff0c;SpringBoot等&#xff0c;这些开发框架提高了我们的开发效率&#xff0c;节省了开发时间。但是这会令我们技术人员处于浮云之上&#xff0c;看不到其本质。说实话&#xff0c;Java语言那么流行&#xff0c;其本…

springboot-自动配置

一、简介 在搭建springboot应用的时候&#xff0c;无需像之前spring的时候&#xff0c;要一堆繁琐的配置文件之类的。一个main的方法&#xff0c;就能把springboot的项目run起来。和其他框架整合也是非常的简单&#xff0c;只需要使用到Enablexxxxx注解就可以搞起来。 二、原理…

外卖项目09---Redis了解

目录 Redis了解 141 一、Redis入门 143 1.1Redis简介 143 1.2Redis下载与安装 143 1.3Redis入门---Redis服务启动 144 1.3.1Redis服务启动 1.3.2设置密码远程连接 二、Redis数据类型 145 三、Redis常用命令 146 3.1Redis常用命令---字符串string操纵命令 3.2Redis…

现代密码学导论-21-分组密码

目录 3.6.3分组密码及其操作模式 ECB(Electronic Code Book) 电码本模式 CBC(Cipher Block Mode) 密文分组链接方式 THEOREM 3.32 CBC模式的CPA安全 连锁CBC模式 OFB(Output Feedback Mode) 输出反馈模式 CTR(Counter) 计数器模式 THEOREM 3.33 CTR多明文PCA安全 THE…

C语言学习之路(基础篇)—— 复合类型(自定义类型)

说明&#xff1a;该篇博客是博主一字一码编写的&#xff0c;实属不易&#xff0c;请尊重原创&#xff0c;谢谢大家&#xff01; 结构体 1) 概述 数组&#xff1a;描述一组具有相同类型数据的有序集合&#xff0c;用于处理大量相同类型的数据运算。 结构体&#xff1a;将多个…

工具分享:Springboot+Netty+Xterm搭建一个网页版的SSH终端

一. 简述 搭建一个web ssh&#xff0c;主要是借助websocket和xterm&#xff0c;可以实现一个类似于xshell的效果&#xff0c;如图&#xff1a; 二. 技术栈 这里使用了springboot、netty、jsch、react、Ts,xterm。 这里我用了springboot和netty实现了websocket&#xff0c;js…

稳了,我用 Python 可视化分析和预测了 2022 年 FIFA 世界杯

许多人称足球为 “不可预测的游戏”&#xff0c;因为一场足球比赛有太多不同的因素可以改变最后的比分。 预测一场比赛的最终比分或赢家确实是很难的&#xff0c;但在预测一项比赛的赢家时就不是这样了。在过去的5年中&#xff0c;Bayern Munich 赢得了所有的德甲联赛&#xf…

如何发布一个属于自己的 npm 包

如何发布一个属于自己的 npm 包 start 在日常的工作中&#xff0c;我们会接触很多 npm 包。 例如&#xff1a; npm install jquerynpm install vue/clinpm install axios# ... 等等有时候会想到&#xff0c;构建一个属于自己的 npm 包&#xff0c;应该超级酷吧&#xff1f; …

zabbix基础环境部署

目录 一、环境准备 二、部署LNMP 1、安装Nginx及其依赖包 2、修改nginx配置 3、测试页面 三、部署zabbix服务端 1、下载zabbix 2、安装源码zabbix 3、为zabbix创建数据库与数据库账户 4、搭建Zabbix页面 4.1、第1步 Check of pre-requisites 4.2、第2步 Configure D…

Python代码的编写运行方式

Python代码的编写运行方式介绍 python官方运行环境可到网站https://www.python.org/downloads/找到合适版本下载安装。 安装比较容易&#xff0c;特别强调&#xff0c;安装过程建议勾选“Add Python to PATH”&#xff08;将Python添加到PATH环境变量&#xff09;【注1】&…

Docker安装Minio

寻找Minio镜像 Docker Hub 查找官方镜像 下载Minio镜像 下载最新版Minio镜像 docker pull minio/minio等同于 docker pull minio/minio:latest下载指定版本的Minio镜像 docker pull minio/minio:RELEASE.2022-11-26T22-43-32Z.fips检查当前所有Docker下载的镜像 docker …

第六章TCP/IP——网络传输硬件设备

个人简介&#xff1a;云计算网络运维专业人员&#xff0c;了解运维知识&#xff0c;掌握TCP/IP协议&#xff0c;每天分享网络运维知识与技能。个人爱好: 编程&#xff0c;打篮球&#xff0c;计算机知识个人名言&#xff1a;海不辞水&#xff0c;故能成其大&#xff1b;山不辞石…

Linux进程通信之消息队列

目录 1.消息队列的原理&#xff1a; 2.消息队列的接口&#xff1a; &#xff08;1&#xff09;创建消息队列 &#xff08;2&#xff09;向消息队列发送消息 &#xff08;3&#xff09;接收消息 &#xff08;4&#xff09;操作消息队列的接口 1.消息队列的原理&#xff1a; 消…

docker 安装 Heimdall 导航页

前言 随着群晖中使用的功能越来越多&#xff0c;各种端口太多容易忘&#xff0c;所以就有了使用导航页的想法&#xff08;使用收藏夹的朋友除外&#xff09;。群晖也有自带的WEB套件管理&#xff0c;不过个人感觉相对麻烦&#xff0c;使用 Heimdall 导航页可以设置密码登录&am…

【数字信号调制】16QAM信号调制解调【含Matlab源码 2050期】

⛄一、QAM调制与解调&#xff08;附实验题目说明&#xff09; 1 引 言 数字振幅调制、数字频率调制和数字相位调制是数字调制的基础,然而,这3种数字调制方式都存在不足之处。如频谱利用率低、抗多径衰落能力差、功率谱衰减慢、带外辐射严重等。为了改善这些不足,几十年来人们不…