3
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

オーディオアセットをダブルクリックするだけでプレビュー再生を行えるエディタ拡張

Last updated at Posted at 2025-08-10

前回、AnimationClipをダブルクリックをするだけでプレビュー再生を行うことが出来るようになるエディタ拡張を公開しました。

「1つ作ると、他のアセット種別でも同様のプレビューを行いたくなる」…という事で、今回はオーディオ(音声)アセットをダブルクリックした際のプレビュー再生を出来るようにします。
いちいち選択→Inspectorから再生するのも面倒だし、かといってダブルクリックしたらWindowsMediaPlayerが開くのもそれはそれで面倒だしね。

コード

早速コードを記載しておきます

AudioClipPreviewWindow.cs
using UnityEditor;
using UnityEditor.Callbacks;
using UnityEngine;
using UnityEngine.Audio;
using UnityEngine.Playables;

namespace ScreenPocket.Editor
{
    /// <summary>
    /// 音声ファイルをダブルクリックしたら別窓で再生をするためのエディタ拡張
    /// </summary>
    public sealed class AudioClipPreviewWindow : EditorWindow
    {
        #region Fields
        /// <summary>
        /// AudioSource保持用オブジェクト
        /// 不可視
        /// </summary>
        private GameObject _previewOwner;
        /// <summary>
        /// 再生用のAudioSource
        /// </summary>
        private AudioSource _previewAudioSource;
        /// <summary>
        /// 再生用のGraph
        /// </summary>
        private PlayableGraph _graph;
        
        /// <summary>
        /// 再生用GraphのOutput。AudioSourceと一対
        /// </summary>
        private AudioPlayableOutput _output;
        
        /// <summary>
        /// 再生したいAudioClip
        /// </summary>
        private AudioClip _previewAudioClip;
        private AudioClipPlayable _audioClipPlayable;
        
        #endregion
        
        /// <summary>
        /// ダブルクリックのコールバック
        /// </summary>
        [OnOpenAsset(0)]
        public static bool OnOpen(int instanceID, int line)
        {
            //AudioClip以外は通常
            if (EditorUtility.InstanceIDToObject(instanceID) is not AudioClip audioClip)
            {
                return false;
            }
            
            Open(audioClip);
            return true;
        }
        
        /// <summary>
        /// メニューから開く場合
        /// この関数は正直不要だけども確認用として残す
        /// </summary>
        [MenuItem("Tools/ScreenPocket/AudioClip Preview")]
        public static void Open()
        {
            Open(null);
        }
        
        /// <summary>
        /// Dropdown形式で開く
        /// </summary>
        /// <param name="clip"></param>
        private static void Open(AudioClip clip)
        {
            var openPosition = Vector2.zero;
            if (Event.current != null)
            {
                //マウスとまったく同じだとちょっと見づらいので少しだけ右下に位置をずらす
                var positionOffset = new Vector2(64f,12f);
                openPosition = Event.current.mousePosition + positionOffset;
            }

            var popupPosition = GUIUtility.GUIToScreenPoint(openPosition);
            var window = CreateInstance<AudioClipPreviewWindow>();
            //横幅は、まぁスライダーが隠れてしまわない程度に
            const float windowWidth = 240f;
            //縦幅は適当に(デフォルト行サイズ+α)*3行分くらい
            var windowHeight = (EditorGUIUtility.singleLineHeight + 4) * 3;
            var windowSize = new Vector2(windowWidth, windowHeight);
            //表示
            window.ShowAsDropDown(new Rect(popupPosition, Vector2.zero), windowSize);
            //渡されたクリップを流し込み
            window._previewAudioClip = clip;
            //同じアセットを何度もダブルクリックした時に備えて時間を0に戻しておく
            window.SetTime(0);
        }

        /// <summary>
        /// 時間を指定
        /// </summary>
        /// <param name="time"></param>
        private void SetTime(double time)
        {
            if (!_audioClipPlayable.IsValid())
            {
                return;
            }
            
            _audioClipPlayable.SetTime(time);
        }

        private void OnEnable()
        {
            _previewOwner = new GameObject("AudioSourceOwnerForPreview")
            {
                hideFlags = HideFlags.HideAndDontSave //非表示化
            };
            _previewAudioSource = _previewOwner.AddComponent<AudioSource>();
            
            _graph = PlayableGraph.Create("AudioClip Preview Window Graph");
            _output = AudioPlayableOutput.Create(_graph, "Output", _previewAudioSource);
            //Outputに未接続だけどとりあえず再生(接続はこの次の UpdateAudioClipPlayable() で行う)
            _graph.Play();
        }

        /// <summary>
        /// 無効時コールバック
        /// 後片付けを行う
        /// </summary>
        private void OnDisable()
        {
            //片付け
            Cleanup();
        }
        
        public void OnGUI()
        {
            //ラベル幅を狭く
            EditorGUIUtility.labelWidth = 60;
            _previewAudioClip = (AudioClip)EditorGUILayout.ObjectField("AudioClip", _previewAudioClip, typeof(AudioClip), false);
            //Clipの状況に応じてPlayableを作り直し
            UpdateAudioClipPlayable();
            //操作部
            OnGuiAudioControl();

            //ラベル幅を戻す
            EditorGUIUtility.labelWidth = 0;
            Repaint();
        }
        
        private void UpdateAudioClipPlayable()
        {
            if (!_graph.IsValid())
            {
                return;
            }

            //すでに再生中?
            if (_audioClipPlayable.IsValid())
            {
                //同じなら抜ける
                if (_audioClipPlayable.GetClip() == _previewAudioClip)
                {
                    return;
                }

                //違うなら削除
                _audioClipPlayable.Destroy();
            }

            //次再生したいアニメが空っぽなら抜ける
            if (_previewAudioClip == null)
            {
                return;
            }

            //Playableを作り直し(確認したいだけなので基本は非ループ)
            _audioClipPlayable = AudioClipPlayable.Create(_graph, _previewAudioClip, looping:false);
            _audioClipPlayable.SetDuration(_previewAudioClip.length);
            //Outputに接続
            _output.SetSourcePlayable(_audioClipPlayable);
        }

        /// <summary>
        /// Audio再生などの操作のGUI
        /// </summary>
        private void OnGuiAudioControl()
        {
            if (!_audioClipPlayable.IsValid())
            {
                return;
            }

            EditorGUI.BeginChangeCheck();
            var time = EditorGUILayout.Slider("Time", (float)_audioClipPlayable.GetTime(), 0f,
                (float)_audioClipPlayable.GetDuration());
            if (EditorGUI.EndChangeCheck())
            {
                SetTime(time);
            }

            EditorGUILayout.BeginHorizontal();
            if (GUILayout.Button("|<<", GUILayout.Width(32)))
            {
                SetTime(0);
            }

            if (GUILayout.Button("||", GUILayout.Width(32)))
            {
                _audioClipPlayable.SetSpeed(0);
            }

            if (GUILayout.Button(">", GUILayout.Width(32)))
            {
                _audioClipPlayable.SetSpeed(1);
            }

            if (GUILayout.Button(">>|", GUILayout.Width(32)))
            {
                SetTime(_audioClipPlayable.GetDuration());
            }

            EditorGUILayout.EndHorizontal();
        }

        /// <summary>
        /// 片付け
        /// </summary>
        private void Cleanup()
        {
            //Playableの削除 GraphのDestroy()でまとめて片付けてくれそうだけども一応
            if (_audioClipPlayable.IsValid())
            {
                _audioClipPlayable.Destroy();
            }

            //Graphの削除
            if (_graph.IsValid())
            {
                _graph.Destroy();
            }

            //再生用オブジェクトの削除
            if (_previewOwner != null)
            {
                if (Application.isPlaying)
                {
                    Destroy(_previewOwner);
                }
                else
                {
                    DestroyImmediate(_previewOwner);
                }
                
                _previewOwner = null;
            }
        }
    }
}

今回のポイントは下記です

  • 今回もPlayableを使用して実装しています
  • OnEnabled()で「再生用のGameObjectを非表示で作成&PlayableGraphを作成」し、OnDisabled()で「全部の片付け」をしています
    • 非表示でGameObjectを作るのは、まぁCinemachineVolumeSettingsもそんな感じでVolume反映しているし良いんじゃないかな?という事で
  • Graphは窓を作ったタイミングで作成&再生しておいて、後から「指定されたAudioClipから作成したPlayable」を紐づけています
    • この部分は前回のコードと同じような処理として残ってしまいましたが、正直GraphとPlayableをまとめて作っちゃっても良いんじゃないかな?という気もしています。
  • Playableは「とりあえず確認できれば良いや」という事で非ループで作成しています。

使用方法

  • AudioClipPreviewWindow.csというファイルを作成し、上のコードをコピペしてください
  • asmdefを使用していて上手くビルドが通らない場合は良い感じにasmdefを調整してください
  • その上で、プロジェクト内にあるmp3やらoggやらをダブルクリックすると小窓が開く想定です。
  • メニューのTools/ScreenPocket/AudioClip Previewを選択しても窓が出ますが、これはおまけなのでお気になさらず

最後に

前回もそうですが、PlayableAPIの取っ掛かりとしては扱いやすいコード量と用途なのではないかな?と考えています。

コードは基本的に自由に改良してご使用ください。
※配布などをされる際は、参考サイトとしてこのページへのリンクを記載したり、私に一報頂けるとありがたいです
Xで私にブロックされている人についてはこの記事に限らず私のQiita記事の一切の利用を禁じます。 該当するかどうかは筆者のXページをご確認ください

自分としてはとりあえず目的は達成できたのでこれ以上触るつもりはないですが、改良するとしたら波形表示やら、スロー、倍速再生、ループ再生切り替えなどを対応しても良さそうですね~

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

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?