Unity Shader 学习(一):初识ShaderLab -- 以“Unlit Shader”模板为例 01

这篇具有很好参考价值的文章主要介绍了Unity Shader 学习(一):初识ShaderLab -- 以“Unlit Shader”模板为例 01。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。


前言

阅读本文前需要对渲染流程有简单了解,并且要知道Unity ShaderLab、Cg/HLSL大概是什么,它们并不是一回事。
然后,看什么教程都不如直接看官方文档:

  • ShaderLab - Unity手册
  • HLSL参考

一、创建Shader

首先创建一个 Unlit Shader 模板(Project 窗口,右键 → Create → Shader → Unlit Shader)
直接上代码:

Shader "Unlit/UnlitShader"
{
    Properties
    {
        _MainTex ("Texture", 2D) = "white" {}
    }
    SubShader
    {
        Tags { "RenderType"="Opaque" }
        LOD 100

        Pass
        {
            CGPROGRAM
            #pragma vertex vert
            #pragma fragment frag
            // make fog work
            #pragma multi_compile_fog

            #include "UnityCG.cginc"

            struct appdata
            {
                float4 vertex : POSITION;
                float2 uv : TEXCOORD0;
            };

            struct v2f
            {
                float2 uv : TEXCOORD0;
                UNITY_FOG_COORDS(1)
                float4 vertex : SV_POSITION;
            };

            sampler2D _MainTex;
            float4 _MainTex_ST;

            v2f vert (appdata v)
            {
                v2f o;
                o.vertex = UnityObjectToClipPos(v.vertex);
                o.uv = TRANSFORM_TEX(v.uv, _MainTex);
                UNITY_TRANSFER_FOG(o,o.vertex);
                return o;
            }

            fixed4 frag (v2f i) : SV_Target
            {
                // sample the texture
                fixed4 col = tex2D(_MainTex, i.uv);
                // apply fog
                UNITY_APPLY_FOG(i.fogCoord, col);
                return col;
            }
            ENDCG
        }
    }
}

官方手册(自定义着色器基础 )大概的讲了一下 。

二、数据类型

1. 属性数据类型

Shader "Unlit/UnlitShader"
{
	Properties
	{
		// 属性     
	}
}
  • 标量类型:

    _IntValue("IntValue", Int) = 1 // 整型
    _FloatValue("FloatValue", Float) = 0.1 // 浮点型(注意:后面不能加 f)
    _Range("Range", Range(0.5, 2)) = 1 // 在某一范围的值(边界值是整型或浮点型都可以)
    

    其实 Int 类型的底层也是 Float 类型Unity Shader 学习(一):初识ShaderLab -- 以“Unlit Shader”模板为例 01,Shader,unity,学习,游戏引擎

  • 向量类型:

    _Color("Color", Color) = (0, 0, 0, 1) // 表示颜色的四维向量 (r, g, b, a),显示取色器
    _Vector("Vector", Vector) = (0, 0, 0, 0) // 四维向量
    
  • 采样器类型:

    _MainTex ("Texture", 2D) = "white" {} // 二维纹理,默认值表示方式为 ""{}
    

    “” 为unity内置纹理,“white”、“black”、“gray”、“bump”、“red”,空字符串或其它字符串表示 “gray”

2. 着色器数据类型(Cg/HLSL)

Shader "Unlit/UnlitShader"
{
    SubShader
    {
        Pass
        {
            CGPROGRAM
            // 变量
			...
            ENDCG
        }
    }
}
  • 标量类型:

    int _IntValue; // 整型
    float _FloatValue; // 浮点型
    float _Range; // 在某一范围的值
    
  • 向量类型:

    float4 _Color; // 颜色
    float4 _Vector; // 四维向量
    
  • 采样器类型:

    sampler2D _MainTex; // 二维纹理采样器
    

表示小数的类型除了 float、half,根据Cg的语法,还有一种精确度:fixed

直接看unity官方文档:着色器数据类型和精度;

Unity Shader 学习(一):初识ShaderLab -- 以“Unlit Shader”模板为例 01,Shader,unity,学习,游戏引擎

总结一下:float:坐标、参与计算; half:向量、HDR颜色; fixed:普通颜色

  • 结构类型:struct,熟悉C系语言的无需多言。
    struct appdata
    {
        float4 vertex : POSITION;
        float2 uv : TEXCOORD0;
    };
    
    struct v2f
    {
        float2 uv : TEXCOORD0;
        // UNITY_FOG_COORDS(1)
        float4 vertex : SV_POSITION;
    };
    

三、语义

参考:
HLSL - 语义
着色器语义 - Unity Manual

1. 语义的作用以及需要语义的时机

  • 时机:在渲染流水线着色器阶段之间传递的所有变量都需要语义;
  • 作用:表明了着色器输入输出的参数的用途

总结一下: 语义就是向渲染流水线每一阶段解释输入或输出的参数表示的含义

类比写一个函数,为该方法添加注释告诉方法的使用者,参数及返回值的含义。

1. 顶点着色器输入

struct appdata
{
    float4 vertex : POSITION;
    float2 uv : TEXCOORD0;
};
  • POSITION对象空间中的顶点位置;
  • TEXCOORD[n]:纹理坐标(TEXCOORD0表示第一套纹理坐标);

2. 片元着色器输入

struct v2f
{
    float2 uv : TEXCOORD0;
    // UNITY_FOG_COORDS(1)
    float4 vertex : SV_POSITION;
};
  • SV_POSITION屏幕空间中的像素中心点位置(参考:HLSL的SV_POSITION);
  • TEXCOORD[n]:纹理坐标(TEXCOORD0表示第一套纹理坐标);

注1:“ SV_ ”,系统值,Direct3D 10 增加,表示该参数由光栅化阶段解释。
注2:像素中心点位置 = 像素位置.xy + (0.5, 0.5)

3. 片元着色器输出

fixed4 frag (v2f i) : SV_Target
{
    // sample the texture
    fixed4 col = tex2D(_MainTex, i.uv);
    // apply fog
    // UNITY_APPLY_FOG(i.fogCoord, col);
    return col;
}

这个语义没有写在结构体参数后面,而是直接写在片元着色器函数后面,其实是因为该函数输出的参数只有一个,可以改写成以下形式,但没必要

struct fo
{
    fixed4 col : SV_Target;
};

fo frag (v2f i)
{
    fo o;
    // sample the texture
    o.col = tex2D(_MainTex, i.uv);
    // apply fog
    // UNITY_APPLY_FOG(i.fogCoord, col);
    return o;
}
  • SV_Target:计算后的像素的颜色。

等一下,还有一个东西没说,不要想糊弄过去:

struct v2f
{
    // float2 uv : TEXCOORD0;
    UNITY_FOG_COORDS(1)
    // float4 vertex : SV_POSITION;
};

这是什么?看着不像是参数啊!

四、CGInclude文件、预处理指令与宏

着色器代码有一行似曾相识:跟C/C++语言引用头文件的写法一模一样;

#include "UnityCG.cginc"

所以,着色器代码中出现一些HLSL标准语法中没有的东西,大概率是unity自定义的,去这个被引用的CGInclude文件UnityCG.cginc里面找,应该就能找到。

文件路径: Unity Editor安装路径 \Editor\Data\CGIncludes

struct v2f
{
    ...
    UNITY_FOG_COORDS(1)
    ...
};

v2f vert (appdata v)
{
    ...
    UNITY_TRANSFER_FOG(o,o.vertex);
    ...
}

fixed4 frag (v2f i) : SV_Target
{
	...
    // apply fog
    UNITY_APPLY_FOG(i.fogCoord, col);
    ...
}

果然,找到了:(不清楚预处理指令的,可以了解一下 预处理器指令 (HLSL) )

Unity Shader 学习(一):初识ShaderLab -- 以“Unlit Shader”模板为例 01,Shader,unity,学习,游戏引擎
Unity Shader 学习(一):初识ShaderLab -- 以“Unlit Shader”模板为例 01,Shader,unity,学习,游戏引擎

这下知道了,在片元着色器输入结构参数中这一行是什么:

struct v2f
{
    ...
    // UNITY_FOG_COORDS(1)
    float fogCoord : TEXCOORD1; // 或这里什么也没有
    ...
};

另外两个同理可得。所以,这些就是Unity定义的雾效相关的在编译器看来就是普普通通的带有语义的参数、函数语句,并不是什么稀奇古怪的语法(暂时不深挖雾效相关源码了,下次一定)。

其它几行,unity官方手册同样有讲: 着色器编译:pragma 指令

#pragma vertex vert
#pragma fragment frag
// make fog work
#pragma multi_compile_fog

比较重要的是这里:

Unity Shader 学习(一):初识ShaderLab -- 以“Unlit Shader”模板为例 01,Shader,unity,学习,游戏引擎


本篇完。下一篇:Unity Shader 学习(二):初识ShaderLab – 以“Unlit Shader”模板为例 02

学习一下HLSL提供的以及UnityCG.cginc文件中一些比较重要的、常用的内置函数。

拓展资料:文章来源地址https://www.toymoban.com/news/detail-775069.html

  • 浮点值与定点值:计算机中的定点数与浮点数
  • Cg语言官方文档:Cg_language - @NVIDIA

到了这里,关于Unity Shader 学习(一):初识ShaderLab -- 以“Unlit Shader”模板为例 01的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 深入URP之Shader篇3: Unlit Shader分析[下]

    上篇中我们分析了Unlit shader的Properties在ShaderGUI中的处理,接下来看Sub Shader。 unlit shader以及其他URP shader包含两个SubShader,分别是针对ShaderModel4.5和2.0。由于unlit shader本身很简单,这两个SubShader几乎一样,唯一的差别是ShaderModel 4.5的SubShader会定义 #pragma multi_compile _ DOTS_INSTANCIN

    2024年02月10日
    浏览(39)
  • 深入URP之Shader篇2: 目录结构和Unlit Shader分析[上]

    我使用的Unity版本为2020.3.33f1,对应的URP和SRP Core版本为10.8.1。阅读URP源码建议把package从Library/PackageCache中拷贝到Packages目录,也就是自定义package的方式,然后推荐使用VS code打开工程,这样可以很方便的跳转代码阅读。 首先,我们看一下URP源码的目录结构,看一下Shader代码的

    2023年04月23日
    浏览(33)
  • 【Linux学习】01Linux初识与安装

    01Linux初识与安装 02Linux基础命令 03Linux用户和权限 04Linux实用操作 05-1Linux上安装部署各类软件 05-2Linux上部署项目 Linux(B站黑马)学习笔记 01Linux初识与安装 硬件和软件 软件:是用户和计算机硬件之间的接口和桥梁,用户通过软件与计算机进行交流。 而操作系统,就是软件的

    2024年02月07日
    浏览(39)
  • Shader学习第七篇:几种Unity的Shader的例子

    下面是几种Shader的例子,从简单到复杂,一步一步了解 Shader 的编写机制。 顶点/片元着色器 Vertex/Fragment Shader,下面我们介绍的示例就是这个。 表面着色器 Surface Shader ,而这个底层Unity也是转成了顶点/片元着色器 固定函数着色器 Fixed Function Shader (已弃用) 在一些低端设备使

    2024年02月09日
    浏览(52)
  • shader学习摘要(九)unity阴影

    在前向渲染中,如果场景中最重要的平行光开启了阴影,unity就会为该光源计算它的阴影映射纹理(shadowmap)。这张阴影映射纹理本质上也是一张深度图,它记录了从该光源的位置出发、能看到的场景中距离它最近的表面位置(深度信息)。 我们通过修改光源的light控件下的

    2024年02月08日
    浏览(49)
  • Unity Shader学习3:透明效果

    Unity中的透明效果由透明通道控制(RGBA中的A),其值为0是完全透明,为1时完全不透明。有两种方法可以实现透明效果: 透明度测试(Alpha Test) 和 透明度混合(Alpha Blend) 。 透明度测试是指通过特定的条件(通常是Alpha通道的值是否超过某个阈值)来判断片元是否透明,只

    2024年01月19日
    浏览(59)
  • Unity Shader学习(九)物体边缘实现

    根据前面的学习,我们了解到除了可以对点的颜色进行处理,还可以对点本身进行操作,例如我们可以改变点的位置,这样就可以实现对模型渲染的操控。物体边缘效果是我们常用的一种效果,要实现物体边缘,原理也很简单。 首先我们要了解到,模型在渲染时,有正面和背

    2024年02月16日
    浏览(48)
  • Unity Shader学习1:基础光照模型

    Phong光照模型 是描述物体的直接光照的简易模型,它认为从物体出发进入摄像机的光由四部分组成: 自发光(emissive),环境光(ambient),漫反射(diffuse),高光(specular)。 c = c e m i s s i v e + c a m b i e n t + c d i f f u s e + c s p e c u l a r c = c_{emissive} + c_{ambient} + c_{diffuse} + c_

    2024年02月03日
    浏览(47)
  • Unity Shader入门精要学习——透明效果

    要么完全透明,要么完全不透明。 实现简单,实质上是一种剔除机制,通过将不满足条件(通常使用小于某个阈值来判定,一般使用clip方法)的片元舍弃的方法来达到完全透明效果。这些被舍弃的片元不会再进行任何的处理,也不会对颜色缓冲产生任何影响,其余满足条件

    2024年02月17日
    浏览(47)
  • UnityShader入门学习(三)——Unity的Shader

    Tags Tags可以写在SubShader的一开始(所有的Pass用),也可以写在Pass块的内部(该Pass用) 渲染设置 同Tags一样也可以写在Pass中或者写在Pass外面(SubShader一开始) Pass介绍 Pass里面可以定义Pass名称 Pass里面的Tags还可以有额外的设置 还有CG语言所写的代码,主要是顶点片元着色器,

    2024年02月16日
    浏览(42)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包