8
4

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 5 years have passed since last update.

Unityで1ポリゴンの波紋を作る

Last updated at Posted at 2019-08-27

この記事について

Unityで下の画像のような波紋(動きます)のようなものを
主にShaderLabを使って作ったので紹介します。

image.png

64bitのWindowsにインストールした
Unity2017.4.28f1、MonoDevelop5.9.6、Metasequoia4Ver4.7.0を使用しています。

1ポリゴンオブジェクトを作る。

とりあえず3Dモデリングソフトで
UV展開が成された正三角形板ポリゴンを
こんな感じに作っていきます。
image.png
image.png

シェーダー

Hamon.shader
Shader "Unlit/Hamon"
{
	Properties
	{
		_MainTex ("Texture", 2D) = "white" {}
		_Power ("Power",Range(0,100)) = 0
		_Period("Period",Range(0,100))=0
		_Speed("Speed",Range(0,1000))=0
		_Detail("Detail",Range(0,100))=0
	}
	SubShader
	{
		Tags { "RenderType"="Opaque"
				"Queue"="Transparent" }
		LOD 100
		GrabPass
		{
			"_BackTexture"
		}
		Cull off
		Pass
		{
			CGPROGRAM
			#pragma vertex vert
			#pragma fragment frag
			// make fog work
			#pragma multi_compile_fog
			
			#include "UnityCG.cginc"

			static const float2 center = float2(0.5,0.5);
			static const float PI = 3.14159265f;

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

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

			sampler2D _MainTex;
			float4 _MainTex_ST;

			sampler2D _BackTexture;
			float _Power;
			float _Period;
			float _Speed;
			float _Detail;
			
			v2f vert (appdata v)
			{
				v2f o;
				o.vertex = UnityObjectToClipPos(v.vertex);
				o.uv = TRANSFORM_TEX(v.uv, _MainTex);
				UNITY_TRANSFER_FOG(o,o.vertex);
				o.screenPos = ComputeGrabScreenPos(o.vertex);
				return o;
			}

			half2 ReturnUV(half2 uv,half2 screenUV)
			{
				//中心座標から現在処理中のuv座標へのベクトル
				float2 c2n = uv - center;
				//c2nの長さ
				float length_of_c2n = sqrt(c2n.x*c2n.x + c2n.y*c2n.y);
				//c2nと平行な単位ベクトル
				float c2n_unit = c2n / length_of_c2n;

				//波
				float nami = (sin(((length_of_c2n * 10 *_Period - _Time*_Speed)%(2*PI))/(length_of_c2n*_Detail))/100 * _Power) * (0.2 - length_of_c2n);

				if(length_of_c2n < 0.2){
				screenUV += c2n_unit * nami;
				screenUV += c2n_unit * nami;
				}
				return screenUV;
			}

			fixed4 frag (v2f i) : SV_Target
			{
				half2 grabUV = i.screenPos.xy / i.screenPos.w;
				// sample the texture
				half2 mainUV = ReturnUV(i.uv,grabUV);
				fixed4 col = tex2D(_BackTexture, mainUV);
				// apply fog
				UNITY_APPLY_FOG(i.fogCoord, col);
				return col;
			}
			ENDCG
		}
	}
}

UnityのUnlitShaderを基に作っています。
初心者なので消していいものが分からないので無駄が多いです。

やってる事としては、
中心のUV座標(0.5,0.5)から
処理中の点のUV座標の
ベクトルを使い計算し、
それを使ってスクリーンのUV座標に影響を与えて
GrabPassのテクスチャを波のように表示させてます。

波の状態を変化させるために
実数のプロパティを何個か作りました。

    Properties
    {
        _MainTex ("Texture", 2D) = "white" {}
        _Power ("Power",Range(0,100)) = 0
        _Period("Period",Range(0,100))=0
        _Speed("Speed",Range(0,1000))=0
        _Detail("Detail",Range(0,100))=0
    }

処理中の点のUV座標とスクリーンのUV座標を引数にとって、
tex2D用のUVを返す関数です。

主にnami変数が働いてます。
UV座標のずらす程度を、
中心から処理中の点へのベクトル と 平行な単位ベクトル
に掛けて波にします。

また、中心からの距離が0.2未満の場合のみ波にして、
中心からの距離が0.2以上の場合は波ではないようにして
1ポリゴンでも円になるようにします。

			half2 ReturnUV(half2 uv,half2 screenUV)
			{
				//中心座標から現在処理中のuv座標へのベクトル
				float2 c2n = uv - center;
				//c2nの長さ
				float length_of_c2n = sqrt(c2n.x*c2n.x + c2n.y*c2n.y);
				//c2nと平行な単位ベクトル
				float c2n_unit = c2n / length_of_c2n;

				//波
				float nami = (sin(((length_of_c2n * 10 *_Period - _Time*_Speed)%(2*PI))/(length_of_c2n*_Detail))/100 * _Power) * (0.2 - length_of_c2n);

				if(length_of_c2n < 0.2){
				screenUV += c2n_unit * nami;
				screenUV += c2n_unit * nami;
				}
				return screenUV;
			}

とりあえず機能はしますが、
もっとコードは綺麗にできるでしょう。

8
4
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
8
4

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?