初めに
この記事は Unity 初学者が学習の備忘録として書いたものです.誤っている点もあるかと思いますので,その際はご指摘いただけると幸いです.
完成イメージ
下図のように 2D ゲームの背景をそれっぽく動かすことを目指しました.
環境:Unity6 Preview (6000.0.4f1)
概要
全体の流れとしては以下の通りです.
- ShaderGraph サンプルを利用してシェーダー作成
- マテリアルを SpriteRender に適用
- プロパティ操作用コンポーネントを作成
- アセット"DOTween"でアニメーションさせる
Shader Graph の公式サンプル
ShaderGraph には、 Procedural Patterns というサンプルデータが提供されています.
詳しくは以下の記事を参照してください.
今回はサンプルの"Stripes"ノードを使用して,各種パラメータを設定できる Unlit シェーダーを作成しました."Stripes"ノードは入力として "Offset" を受け付けているため,ここに "Time"ノードの出力を入れることで時間経過によるスクロールが可能です.
※"Time"ノードと UV オフセットを使用すれば容易に動きのあるシェーダーを作成できます.
下表のプロパティを作成しました.プロパティは Graph Editor のブラックボードで作成できます.
※このあと C#スクリプトからアクセスする際には,下表の [Reference 名] でプロパティにアクセスします.
Name | Reference | 型 | 役割 |
---|---|---|---|
Frequency | _Frequency | Float | 周期(線の本数) |
Thickness | _Thickness | Float | 線の間隔 |
ScrollX | _ScrollX | Float | X 方向のオフセット速度 |
ScrollY | _ScrollY | Float | Y 方向のオフセット速度 |
Rotation | _Rotation | Float | 線の傾き |
Color1 | _Color1 | Color | 色 |
Color2 | _Color2 | Color | 色 |
背景オブジェクトの準備
背景は Sprite Renderer で作成することにします.
Create GameObject Menu の "2D Object > Sprites > Square" でオブジェクトを作成して,Material 項目に作成したマテリアルを適用します.
マテリアルの操作に関して
プロパティの設定
マテリアルのプロパティは "SetFloat", "SetVector" 等のセッターで設定できます.引数に [プロパティ名],もしくは[プロパティ ID] を渡すことで,任意のプロパティを指定できます.
Material _material = GetComponent<Renderer>().material;
// 名前指定で_Colorプロパティを設定
_material.SetColor("_Color", Color.white);
// ID指定で_Colorプロパティを設定
int colorPropID = Shader.PropertyToID("_Color");
_material.SetColor(colorPropID, Color.white);
↓ Shader がどのようなプロパティを持っているかはインスペクタの"Properities"項目で確認することができます.
マテリアルの破棄
またマテリアルのプロパティにアクセスすると,自動でコピーが生成されます(プロパティの変更はコピーに対して行われる).そのためコピーされたオブジェクトは自分で破棄しないとメモリリークを引き起こすようです.
Material _material;
private void Awake() {
_material = GetComponent<Renderer>().material;
_material.color = Color;
}
private void OnDestroy() {
if (_material != null) {
Destroy(_material); // 自分で破棄する
_material = null;
}
}
Resources.UnloadUnusedAssets() でも使われてないマテリアルを削除することができます.
適当な操作コンポーネントを作成
必須ではありませんが,materal.GetVelue() / materal.SetVelue()の操作を簡略化するため下コードのようにアクセス用のプロパティを用意しておきます.
[RequireComponent(typeof(SpriteRenderer))]
public class Test : MonoBehaviour {
private Material _material;
// ※簡易操作用のProperity
public float Frequency {
get => _material.GetFloat("_Frequency");
set => _material.SetFloat("_Frequency", value);
}
private void Awake() {
_material = gameObject.GetComponent<SpriteRenderer>().material;
}
private void OnDestroy() {
Destroy(_material);
}
}
さほどメリットを感じませんが一応マテリアルを意識せずにパラメータ設定が行えます.
[RequireComponent(typeof(SpriteRenderer))]
public class Test : MonoBehaviour {
// インスペクタでここを操作すると,変動することを確認できる
[Range(0, 30)] public float frequency;
private void Update() {
Frequency = frequency;
}
}
またアセット"DOTween"には任意のプロパティをアニメーションさせる DOTween.To() というメソッドがあります.こちらを用いると簡単にアニメーション処理を実装できます.
[RequireComponent(typeof(SpriteRenderer))]
public class Test : MonoBehaviour {
// "Frequency"を指定した値までアニメーションさせる関数
public Tweener DOFloat_Frequency(float endValue, float duration) {
return DOTween.To(
() => Frequency,
x => Frequency = x,
endValue,
duration
).SetLink(gameObject);
}
IEnumerator Start() {
while (true) {
if (Input.GetKeyDown(KeyCode.Return)) {
// アニメーション開始
DOFloat_Frequency(endValue: 100, duration: 1);
break;
}
yield return null;
}
}
}
DOTween の文法については以下の記事を参照ください.
最終的なコード
Material Handler (マテリアル操作を簡略化するためのラッパー)
※マテリアルは動的に生成する形に変更しています.using System;
using UnityEngine;
namespace nitou {
/// <summary>
/// マテリアルのプロパティ操作用ラッパークラス
/// </summary>
public abstract class MaterialHandler : IDisposable {
protected readonly Shader _shader = null;
protected Material _material = null;
/// ----------------------------------------------------------------------------
// Public Method
public MaterialHandler(Shader shader) {
if (shader == null) throw new ArgumentNullException(nameof(shader));
// マテリアル生成
_shader = shader;
_material = new Material(_shader);
DefinePropertyID();
}
public void Dispose() {
if (_material == null) return;
GameObject.Destroy(_material);
_material = null;
}
/// <summary>
/// レンダラーにマテリアルを適用する
/// </summary>
public void ApplayMaterial(Renderer renderer) {
if (renderer == null) throw new ArgumentNullException(nameof(renderer));
renderer.sharedMaterial = _material;
}
/// ----------------------------------------------------------------------------
// Protected Method
/// <summary>
/// マテリアルのプロパティID定義
/// </summary>
protected abstract void DefinePropertyID();
}
public static partial class RendererExtensions {
/// <summary>
/// レンダラーにマテリアルを適用する拡張メソッド
/// </summary>
public static void SetSharedMaterial(this Renderer self, MaterialHandler handler) {
handler.ApplayMaterial(self);
}
}
}
↓ 今回作ったシェーダー用のクラス.
using UnityEngine;
namespace nitou {
public class StripeMaterial : MaterialHandler {
// ID
protected int _frequencyID;
protected int _thicknessID;
protected int _rotationID;
protected int _scrollXID;
protected int _scrollYID;
protected int _color1Id;
protected int _color2Id;
/// ----------------------------------------------------------------------------
// Properity
public float Frequency {
get => _material.GetFloat(_frequencyID);
set => _material.SetFloat(_frequencyID, value);
}
public float Thickness {
get => _material.GetFloat(_thicknessID);
set => _material.SetFloat(_thicknessID, value);
}
public float Rotation {
get => _material.GetFloat(_rotationID);
set => _material.SetFloat(_rotationID, value);
}
public float ScrollX {
get => _material.GetFloat(_scrollXID);
set => _material.SetFloat(_scrollXID, value);
}
public float ScrollY {
get => _material.GetFloat(_scrollYID);
set => _material.SetFloat(_scrollYID, value);
}
public Color Color1 {
get => _material.GetColor(_color1Id);
set => _material.SetColor(_color1Id, value);
}
public Color Color2 {
get => _material.GetColor(_color2Id);
set => _material.SetColor(_color2Id, value);
}
/// ----------------------------------------------------------------------------
// Public Method
public StripeMaterial(Shader shader) : base(shader) {}
/// ----------------------------------------------------------------------------
// Protected Method
/// <summary>
/// マテリアルのプロパティID定義
/// </summary>
protected override void DefinePropertyID() {
_frequencyID = Shader.PropertyToID("_Frequency");
_thicknessID = Shader.PropertyToID("_Thickness");
_rotationID = Shader.PropertyToID("_Rotation");
_scrollXID = Shader.PropertyToID("_ScrollX");
_scrollYID = Shader.PropertyToID("_ScrollY");
_color1Id = Shader.PropertyToID("_Color1");
_color2Id = Shader.PropertyToID("_Color2");
}
}
}
Material Controller (Rendererを持つGameObjectにアタッチするコンポーネント)
using UnityEngine;
namespace nitou {
/// <summary>
/// マテリアルの操作を行うコンポーネント
/// </summary>
public abstract class MaterialController<T> : MonoBehaviour
where T : MaterialHandler {
[SerializeField] Renderer _renderer = null;
[SerializeField] Shader _shader = null;
protected T _handler = null;
/// ----------------------------------------------------------------------------
// MonoBehaviour Method
private void Awake() {
_handler = CreateHandler(_shader);
_renderer.SetSharedMaterial(_handler);
}
private void OnDestroy() {
_handler?.Dispose();
}
/// ----------------------------------------------------------------------------
// Protected Method
/// <summary>
/// シェーダーからマテリアルとハンドラを生成する
/// </summary>
protected abstract T CreateHandler(Shader shader);
#if UNITY_EDITOR
private void OnValidate() {
if (_renderer == null) _renderer = gameObject.GetComponent<Renderer>();
}
#endif
}
}
今回作ったシェーダーのマテリアルを操作するコンポーネント
using UnityEngine;
using DG.Tweening;
namespace nitou {
public class StripeMaterialController : MaterialController<StripeMaterial> {
/// ----------------------------------------------------------------------------
// Protected Method
protected override StripeMaterial CreateHandler(Shader shader) {
return new StripeMaterial(shader);
}
/// ----------------------------------------------------------------------------
// Public Method
public Tweener Tween_Frequency(float endValue, float duration) {
return DOTween.To(
() => _handler.Frequency,
x => _handler.Frequency = x,
endValue,
duration
).SetLink(gameObject);
}
public Tweener Tween_Thickness(float endValue, float duration) {
return DOTween.To(
() => _handler.Thickness,
x => _handler.Thickness = x,
endValue,
duration
).SetLink(gameObject);
}
public Tweener Tween_Rotation(float endValue, float duration) {
return DOTween.To(
() => _handler.Rotation,
x => _handler.Rotation = x,
endValue,
duration
).SetLink(gameObject);
}
public Tweener Tween_ScrollX(float endValue, float duration) {
return DOTween.To(
() => _handler.ScrollX,
x => _handler.ScrollX = x,
endValue,
duration
).SetLink(gameObject);
}
public Tweener Tween_ScrollY(float endValue, float duration) {
return DOTween.To(
() => _handler.ScrollY,
x => _handler.ScrollY = x,
endValue,
duration
).SetLink(gameObject);
}
}
}
以下のクラスで簡単な動作確認をした結果です.(今回作ったシェーダーではアスペクト比の変更に対応していないので,変形させると斜線が崩れます.)
// 動作確認用
public class testMain : MonoBehaviour {
[SerializeField] StripeMaterialController _controller;
private Tween _tween;
void Update() {
if (Input.GetKeyDown(KeyCode.J)) {
_tween?.Kill();
_tween = TweenA(1);
} else if (Input.GetKeyDown(KeyCode.K)) {
_tween?.Kill();
_tween = TweenB(1);
}
}
public Tween TweenA(float duration) {
return DOTween.Sequence()
.Join(_controller.Tween_Frequency(20f, duration))
.Join(_controller.Tween_Thickness(-0.1f, duration))
.Join(_controller.Tween_Rotation(-45f, duration))
// スクロール速度・方向
.Join(_controller.Tween_ScrollX(0, 0.2f))
.Join(_controller.Tween_ScrollY(0, 0.2f));
}
public Tween TweenB(float duration) {
return DOTween.Sequence()
.Join(_controller.Tween_Frequency(10f, duration))
.Join(_controller.Tween_Thickness(0.5f, duration))
.Join(_controller.Tween_Rotation(45f, duration))
// スクロール速度・方向
.Join(_controller.Tween_ScrollX(0.5f, 0.2f))
.Join(_controller.Tween_ScrollY(0.5f, 0.2f));
}
終わりに
Procedural Patterns には様々なサンプルデータが含まれるため,色々なパターンを試せそうです.