UnityShader(十四)纹理

目录

前言:

单张纹理实现效果:

效果:


前言:

纹理最初的目的是用一张图片来控制模型的外观。使用纹理映射技术我们可以把一张图"贴"在模型表面,逐纹素(文素的名字是为了和像素进行区分)控制模型颜色。

在建模时候,通常会在建模软件中利用纹理展开技术把纹理映射坐标 存储到每个顶点上,通常这些坐标用一个二维变量**(u,v)** 表示,其中u 为横坐标,v为纵坐标,纹理映射坐标也叫做。

单张纹理实现效果:

采用BlinnPhong光照模型效果:

复制代码
Shader "MyShader/Texture"
{
    Properties
    {
        _MainTex("主贴图",2D)="white"{}
        _MainColor("主颜色",color)=(1,1,1,1)
        _SpecularColor("高光颜色",color)=(1,1,1,1)
        _Gloss("高光范围",float)=1.0
    }
    SubShader
    {
        Pass
        {
            Tags{"LigthMode"="ForwardBase"}

            CGPROGRAM
            #pragma vertex vert 
            #pragma fragment frag 
            #include "UnityCG.cginc"
            #include "Lighting.cginc"

            sampler2D _MainTex;
            float4 _MainTex_ST;
            fixed4 _MainColor;
            fixed4 _SpecularColor;
            float _Gloss;

            struct vertexInput
            {
                float4 vertex:POSITION;
                float3 normal:NORMAL;
                float4 texcood:TEXCOORD0;
            };

            struct vertexOutput
            {
                float4 pos:SV_POSITION;
                float3 worldNormal:TEXCOORD0;
                float3 worldPos:TEXCOORD1;
                float2 uv:TEXCOORD2;
            };

            vertexOutput vert(vertexInput v)
            {
                vertexOutput o = (vertexOutput)o;
                o.pos = UnityObjectToClipPos(v.vertex);
                o.worldNormal = UnityObjectToWorldNormal(v.normal);
                o.worldPos = mul(unity_ObjectToWorld,v.vertex).xyz;
                o.uv = TRANSFORM_TEX(v.vertex,_MainTex);
                return o;
            }

            float4 frag(vertexOutput i):SV_TARGET
            {
                half3 worldNormal = normalize(i.worldNormal);
                half3 worldLightDir = normalize(UnityWorldSpaceLightDir(i.worldPos));
                half3 viewDir = normalize(UnityWorldSpaceViewDir(i.worldPos));
                //half3 reflectDir = normalize(reflect(-worldLightDir,worldNormal));
                half3 halfDir = normalize(viewDir+worldLightDir);
                fixed3 albedo = tex2D(_MainTex,i.uv).rgb*_MainColor.rgb;

                fixed3 diffuse = albedo*_LightColor0.rgb*saturate(dot(worldNormal,worldLightDir));
                fixed3 ambient = UNITY_LIGHTMODEL_AMBIENT.xyz*albedo;
                fixed3 specular = _LightColor0.rgb*_SpecularColor.rgb*pow(saturate(dot(worldNormal,halfDir)),_Gloss);
                fixed3 color = diffuse+ambient+specular;
                return float4(color,1.0);
            }
            ENDCG
        }
    }
}

效果:

在我们声明的变量中有一个_MainTex_ST,这一个变量不是随意定义的,其中S和T分别代表缩放和偏移。在编辑面板中我们可以看到主贴图旁边有一个四位参数变量,如下:

主帖图参数名_ST

指代的就是这个四维变量,其中xy代表缩放,zw代表偏移,在代码中

o.uv = TRANSFORM_TEX(v.texcoord,_MainTex)

实现的就是通过这个四维变量对贴图进行缩放和偏移的功能,当然TRANSFORM_TEX()是Unity封装的方法,其原理就是对贴图的uv坐标先进行xy的缩放,在进行zw的偏移,我们也可以写成:

o.uv = v.vexcoord.xy*_MainTex_ST.xy+_MainTex_ST.ze;

相关推荐
黄思搏21 小时前
基于标注平台数据的 Unity UI 自动化构建工作流设计与工程实践
ui·unity·蓝湖·vectoui
2301_822703201 天前
开源鸿蒙跨平台Flutter开发:跨端图形渲染引擎的类型边界与命名空间陷阱:以多维雷达图绘制中的 dart:ui 及 StrokeJoin 异常为例
算法·flutter·ui·开源·图形渲染·harmonyos·鸿蒙
AI_零食1 天前
Flutter 框架跨平台鸿蒙开发 - 社交断舍离应用
运维·服务器·学习·flutter·游戏·开源·harmonyos
charlie1145141911 天前
通用GUI编程技术——图形渲染实战(二十五)——Alpha混合与透明效果:分层窗口实战
c++·windows·学习·图形渲染·win32
charlie1145141911 天前
通用GUI编程技术——图形渲染实战(二十四)——GDI Region与裁切:不规则窗口与可视化控制
c++·windows·学习·c·图形渲染·win32
雷焰财经1 天前
首都在线MaaS平台:打造企业级AI中枢,驱动游戏产业智变
大数据·人工智能·游戏
wanhengidc1 天前
云手机 热血传奇游戏挂机
服务器·网络·安全·游戏·智能手机
羊羊20351 天前
开发手札:Unity6000与Android交互
android·unity·android-studio
黑客说1 天前
AI 游戏:从固定剧本到无限宇宙
人工智能·游戏
nhc0882 天前
贵阳纳海川·花卉游戏行业解决方案
人工智能·游戏·微信小程序·软件开发·小程序开发