Как прописать поворот текстуры в Unity-шейдере?

Имеется шейдер поставляемый Tiled2Unity:
// Upgrade NOTE: upgraded instancing buffer 'MyProperties' to new syntax.

Shader "Tiled2Unity/Default (Instanced)"
{
    Properties
    {
        _MainTex ("Tiled Texture", 2D) = "white" {}
        _Color ("Tint", Color) = (1,1,1,1)
        [MaterialToggle] PixelSnap ("Pixel snap", Float) = 0
    }

    SubShader
    {
        Tags
        { 
            "Queue"="Transparent"
            "IgnoreProjector"="True"
            "RenderType"="Transparent"
            "PreviewType"="Plane"
        }

        Cull Off
        Lighting Off
        ZWrite Off
        Fog { Mode Off }
        Blend SrcAlpha OneMinusSrcAlpha

        Pass
        {
        CGPROGRAM
            #pragma vertex vert
            #pragma fragment frag
            #pragma multi_compile DUMMY PIXELSNAP_ON
            #include "UnityCG.cginc"
            #include "Tiled2Unity.cginc"

            struct appdata_t
            {
                float4 vertex   : POSITION;
                float4 color    : COLOR;
                float2 texcoord : TEXCOORD0;
                UNITY_VERTEX_INPUT_INSTANCE_ID
            };

            struct v2f
            {
                float4 vertex   : SV_POSITION;
                fixed4 color    : COLOR;
                half2 texcoord  : TEXCOORD0;
            };


            UNITY_INSTANCING_BUFFER_START(MyProperties)
                UNITY_DEFINE_INSTANCED_PROP(float4, _Color)
#define _Color_arr MyProperties
            UNITY_INSTANCING_BUFFER_END(MyProperties)

            v2f vert(appdata_t In)
            {
                UNITY_SETUP_INSTANCE_ID(In);

                v2f Out;
                Out.vertex = UnityObjectToClipPos(In.vertex);
                Out.texcoord = In.texcoord;
                Out.color = In.color * UNITY_ACCESS_INSTANCED_PROP(_Color_arr, _Color);

                #ifdef PIXELSNAP_ON
                Out.vertex = UnityPixelSnap (Out.vertex);
                #endif

                return Out;
            }

            sampler2D _MainTex;

            fixed4 frag(v2f In) : COLOR
            {
                half4 texcol = tex2D(_MainTex, In.texcoord);
                texcol = texcol * In.color;
                return texcol;
            }
        ENDCG
        }
    }

    Fallback "Sprites/Default"
}


Вот ренден:
5aa6cf133a836110555941.png

Как реализовать случайный rotate на (0, 90, 180, 270 градусов) для материала? Хочу чтоб спрайт травы не был однообразным.
  • Вопрос задан
  • 1257 просмотров
Решения вопроса 1
BasmanovDaniil
@BasmanovDaniil
Геймдизайнер-телепат
Для вашей проблемы есть несколько решений. Самое простое - добавить в атлас три повёрнутых картинки травы и использовать их в кисти, которая рисует случайными тайлами. Так вы в будущем сможете перерисовать графику без пересборки карты и вам не придётся поддерживать специальные шейдеры или руками вращать что-то на сцене.

Второе по простоте решение - воспользоваться возможностями Tiled и вращать тайлы там, Tiled это умеет, Tiled2Unity тоже, скорее всего.

Если же прям сильно хотите шейдером, то можно, например, с помощью двух флагов поворачивать uv во все четыре стороны.
Custom/RotationExample
Shader "Custom/RotationExample"
{
    Properties
    {
        _MainTex("Texture", 2D) = "white" {}

        // Волшебные свойства для инспектора
        [Toggle] _RotateCW90("Rotate Clockwise 90", Float) = 0.0
        [Toggle] _Rotate180("Rotate 180", Float) = 0.0
        ////////////////////////////////////
    }
    SubShader
    {
        Tags { "RenderType"="Opaque" "PreviewType" = "Plane" }
        LOD 100

        Pass
        {
            CGPROGRAM
            #pragma vertex vert
            #pragma fragment frag

            // Флаги для компиляции нескольких вариантов шейдера
            #pragma multi_compile __ _ROTATECW90_ON
            #pragma multi_compile __ _ROTATE180_ON
            ////////////////////////////////////////////////////
            
            #include "UnityCG.cginc"

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

            struct v2f
            {
                float4 vertex : SV_POSITION;
                float2 uv : TEXCOORD0;
            };

            sampler2D _MainTex;
            float4 _MainTex_ST;
            
            v2f vert (appdata v)
            {
                v2f o;
                o.vertex = UnityObjectToClipPos(v.vertex);
                o.uv = TRANSFORM_TEX(v.uv, _MainTex);
                return o;
            }
            
            fixed4 frag (v2f i) : SV_Target
            {

                // Собственно код вращения uv
#if _ROTATECW90_ON
                i.uv = float2(1 - i.uv.y, i.uv.x);
#endif
#if _ROTATE180_ON
                i.uv = float2(1 - i.uv.x, 1 - i.uv.y);
#endif
                /////////////////////////////

                fixed4 color = tex2D(_MainTex, i.uv);
                return color;
            }
            ENDCG
        }
    }
}

Я комментариями пометил три важных кусочка, которые можно засунуть в любой шейдер для вращения текстуры. Код выше работает только для квадов с простейшей квадратной uv-картой, растянутой на весь диапазон от нуля до единицы, для атласов надо придумывать другое решение. Ну и не забывайте, что эти флаги надо будет как-то выставлять.
Ответ написан
Пригласить эксперта
Ваш ответ на вопрос

Войдите, чтобы написать ответ

Похожие вопросы