LoginSignup
3
4

More than 3 years have passed since last update.

VisualEffectGraphのPropertyBinderを自作する

Last updated at Posted at 2021-02-09

初めに

UnityのVisualEffectGraphには任意のゲームオブジェクトの位置や入力情報などを取得してPropertyに格納できるPropertyBinderという便利な機能があります。
今回はAudioSourceで再生している音声の波形をVFXGraphに渡すPropertyBinderを自作してみます。デフォルトでもAudioSourceのスペクトラムを渡すBindingが存在しますが音声の波形を渡す機能が無いので、この2つの波形を渡すBindingを自作します。
自作したPropertyBinderを利用したサンプルプログラムは ここ にあります。

作成方法

VFXBinderBaseというクラスを継承した子クラスを作成し、IsValidメソッドとUpdateBindingメソッドを実装することでBindingとして利用可能になります。
VFXBinder属性は実際にこのPropertyBindingsを追加する際のパスとなります。

using UnityEngine;
using UnityEngine.VFX;
using UnityEngine.VFX.Utility;

[VFXBinder("sh00t/AudioWave")]
public class AudioBinder : VFXBinderBase
{
    [VFXPropertyBinding("UnityEngine.Texture2D"), SerializeField]
    ExposedProperty outputWaveMapProperty = "OutputWaveMap";

    [VFXPropertyBinding("UnityEngine.Texture2D"), SerializeField]
    ExposedProperty outputSpectrumMapProperty = "OutputSpectrumMap";

    [VFXPropertyBinding("System.UInt32"), SerializeField]
    ExposedProperty waveLengthProperty = "WaveLength";

    [VFXPropertyBinding("System.UInt32"), SerializeField]
    ExposedProperty spectrumLengthProperty = "SpectrumLength";

    [SerializeField]
    private AudioSource audioSource;
    private Color[] outputColors;
    private Color[] outputSpectrumColors;
    private Texture2D outputWaveMap;
    private Texture2D outputSpectrumMap;
    private float[] outputWave;
    private float[] outputSpectrum;
    public int waveLength = 512;
    public int spectrumLength = 512;
    public FFTWindow fftWindow = FFTWindow.BlackmanHarris;

    public override bool IsValid(VisualEffect component)
    {
        return audioSource != null &&
            component.HasTexture(outputWaveMapProperty) &&
            component.HasTexture(outputSpectrumMapProperty) &&
            component.HasUInt(waveLengthProperty) &&
            component.HasUInt(spectrumLengthProperty);
    }

    private void GetWave()
    {
        if (outputWave == null || waveLength != outputWave.Length)
        {
            outputWave = new float[waveLength];
            outputColors = new Color[waveLength];

            outputWaveMap = new Texture2D(waveLength, 1, TextureFormat.RFloat, false);
            outputWaveMap.filterMode = FilterMode.Bilinear;
            outputWaveMap.wrapMode = TextureWrapMode.Clamp;
        }

        if (outputSpectrum == null || spectrumLength != outputSpectrum.Length)
        {
            outputSpectrum = new float[spectrumLength];
            outputSpectrumColors = new Color[spectrumLength];

            outputSpectrumMap = new Texture2D(spectrumLength, 1, TextureFormat.RFloat, false);
            outputSpectrumMap.filterMode = FilterMode.Bilinear;
            outputSpectrumMap.wrapMode = TextureWrapMode.Clamp;
        }

        audioSource.GetOutputData(outputWave, 1);

        for (var i = 0; i < waveLength; i++)
        {
            outputColors[i].r = outputWave[i];
        }
        outputWaveMap.SetPixels(outputColors);
        outputWaveMap.Apply();

        audioSource.GetSpectrumData(outputSpectrum, 1, fftWindow);
        for (var i = 0; i < spectrumLength; i++)
        {
            outputSpectrumColors[i].r = outputSpectrum[i];
        }
        outputSpectrumMap.SetPixels(outputSpectrumColors);
        outputSpectrumMap.Apply();
    }

    public override void UpdateBinding(VisualEffect component)
    {
        GetWave();

        component.SetTexture(outputWaveMapProperty, outputWaveMap);
        component.SetTexture(outputSpectrumMapProperty, outputSpectrumMap);
        component.SetUInt(waveLengthProperty, (uint)waveLength);
        component.SetUInt(spectrumLengthProperty, (uint)spectrumLength);
    }
}

IsValidはバインディングが作成できるかどうかを検証するメソッドです。このメソッドがtrueを返した場合のみ後述するUpdateBindingメソッドが実行されます。このメソッドによって有効ではないと判断されると警告が表示されます
VFXGraphAudioBinder - SampleScene - PC, Mac & Linux Standalone - Unity 2020.2.1f1 Personal_ _DX11_ 2021_02_10 1_57_58 (2).pngVFXGraphAudioBinder - SampleScene - PC, Mac & Linux Standalone - Unity 2020.2.1f1 Personal_ _DX11_ 2021_02_10 1_58_05 (2).png
今回はAudioSourceの登録と、各ExposedPropertyに存在するプロパティが入力されていない場合にfalseを返す処理にしました。
VFXPropertyBinding属性は一致する型をVFXGraphのPropertyから選択できるドロップダウンを提供します。
vfx (2).pngVFXGraphAudioBinder - SampleScene - PC, Mac & Linux Standalone - Unity 2020.2.1f1 Personal_ _DX11_ 2021_02_10 2_13_01 (2).png
UpdateBindingメソッドではIsValidメソッドがtrueを返した時に行われる、プロパティに値を渡す処理を作成します。音声波形とスペクトラムを取得してプロパティに渡すテクスチャを更新しています。初回実行時と途中で取得する波形の長さが変更された場合には新しく配列などを生成し直しています。

終わりに

PropertyBinderを自作することでいろんな情報をインスペクター上でVFXGraphのプロパティと結びつけられるのでどんどん使っていきましょう。
gif_animation_002.gif

参考

3
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
3
4