98835c3936
* Barebones title screen prefab added * logo and stuff * cool * Added sfx to title screen * The logo now bops to the beat * epic reveal * Fixed something * put some of the stuff into the main menu * other logo bop * Implemented logobop2 and starbop * added scrolling bg, tweaked positioning and wip splash text for play button * more menu * ooops * Expand implemented * cool * Made stars spawn in in the opening * make UI elements look nicer on different aspect ratios * add sound while hovering over logo * add settings menu to title screen make the title screen properly play after the opening * swap out title screen hover sound remove the old config path warning * every button works, some play mode fixes * fix issues with beataction/multisound and pausing * fix dropdown menus not working in certain screens * fix particles rotating when camera controls are used * touch style pause menu items only trigger if cursor is over an item * various changes make playback (unpausing) more reliable only apply changes to advanced audio settings on launch fix title screen visuals add opening music continue past opening by pressing a key update credits * almost forgot this * lol * initial flow mems * user-taggable fonts in textboxes * alt materials for kurokane * assets prep * plan out judgement screen layout change sound encodings * start sequencing judgement * judgement screen sequence * full game loop * fix major issue with pooled sound objects rebalance ranking audio fix issues with some effects in play mode * new graphics * particles * make certain uses of the beat never go below 0 fix loop of superb music * make markers non clamped lockstep frees rendertextures when unloading * lockstep creates its own rendertextures swapped button order on title screen added null checks to animation helpers disabled controller auto-search for now * enable particles on OK rank * play mode info panel * let play mode handle its own fade out * fix that alignment bug in controller settings * more safety here * Update PauseMenu.cs * settable (one-liner) rating screen text * address minigame loading crashes * don't do this twice * wav converter for mp3 * Update Minigames.cs * don't double-embed the converted audio * studio dance bugfixing spree * import redone sprites for studio dance * update jukebox prep epilogue screen * epilogue screen * studio dance inkling shuffle test * new studio dance choreo system * markers upgrade * fix deleting volume changes and markers prep category markers * Update Editor.unity * new rating / epilogue settings look * update to use new tooltip system mark certain editor components as blocking * finish category system * dedicated tempo / volume marker dialogs * swing prep * open properties dialog if mapper hasn't opened it prior for this chart fix memory copy bug when making new chart * fix ctrl + s * return to title screen button * make graphy work everywhere studio dance selector membillion mems * make sure riq cache is clear when loading chart * lol * fix the stupid * bring back tempo and volume change scrolling * new look for panels * adjust alignment * round tooltip * alignment of chart property prefab * change scale factor of mem * adjust open captions material no dotted BG in results commentary (only epilogue) bugfix for tempo / volume scroll * format line 2 of judgement a bit better update font * oops * adjust look of judgement score bar * new rating bar * judgement size adjustment * fix timing window scaling with song pitch * proper clamping in dialogs better sync conductor to dsptime (experiment) * disable timeline pitch change when song is paused enable perfect challenge if no marker is set to do so * new app icon * timing window values are actually double now * improve deferred timekeep even more * re-generate font atlases new app icon in credits * default epilogue images * more timing window adjustment * fix timing display when pitched * use proper terminology here * new logo on titlescreen * remove wip from play update credits * adjust spacing of play mode panel * redo button spacing * can pass title screen with any controller fix issues with controller auto-search * button scale fixes * controller title screen nav * remove song genre from properties editor * disable circle cursor when not using touch style * proper selection graphic remove refs re-add heart to the opening * controller support in opening --------- Co-authored-by: ev <85412919+evdial@users.noreply.github.com> Co-authored-by: minenice55 <star.elementa@gmail.com> Co-authored-by: ThatZeoMan <67521686+ThatZeoMan@users.noreply.github.com>
326 lines
12 KiB
C#
326 lines
12 KiB
C#
using System;
|
|
using System.Collections;
|
|
using System.Collections.Generic;
|
|
using UnityEngine;
|
|
|
|
namespace HeavenStudio.Util
|
|
{
|
|
public class SoundByte
|
|
{
|
|
static GameObject oneShotAudioSourceObject;
|
|
static AudioSource oneShotAudioSource;
|
|
static int soundIdx = 0;
|
|
|
|
public enum AudioType
|
|
{
|
|
OGG,
|
|
WAV
|
|
}
|
|
|
|
public static Sound GetAvailableScheduledSound()
|
|
{
|
|
// soundIdx++;
|
|
// soundIdx %= GameManager.instance.SoundObjects.Count;
|
|
// GameManager.instance.SoundObjects[soundIdx].Stop();
|
|
// return GameManager.instance.SoundObjects[soundIdx];
|
|
|
|
return GameManager.instance.SoundObjects.Get();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Ensures that the jukebox and one-shot audio source exist.
|
|
/// </summary>
|
|
public static void BasicCheck()
|
|
{
|
|
if (FindJukebox() == null)
|
|
{
|
|
GameObject Jukebox = new GameObject("Jukebox");
|
|
Jukebox.AddComponent<AudioSource>();
|
|
Jukebox.tag = "Jukebox";
|
|
}
|
|
if (oneShotAudioSourceObject == null)
|
|
{
|
|
oneShotAudioSourceObject = new GameObject("OneShot Audio Source");
|
|
oneShotAudioSource = oneShotAudioSourceObject.AddComponent<AudioSource>();
|
|
UnityEngine.Object.DontDestroyOnLoad(oneShotAudioSourceObject);
|
|
}
|
|
}
|
|
|
|
public static GameObject FindJukebox()
|
|
{
|
|
if (GameObject.FindGameObjectWithTag("Jukebox") != null)
|
|
return GameObject.FindGameObjectWithTag("Jukebox");
|
|
else
|
|
return null;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Stops all currently playing sounds.
|
|
/// </summary>
|
|
public static void KillOneShots()
|
|
{
|
|
if (oneShotAudioSource != null)
|
|
{
|
|
oneShotAudioSource.Stop();
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Pauses all currently playing sounds.
|
|
/// </summary>
|
|
public static void PauseOneShots()
|
|
{
|
|
if (oneShotAudioSource != null)
|
|
{
|
|
oneShotAudioSource.Pause();
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Unpauses all currently playing sounds.
|
|
/// </summary>
|
|
public static void UnpauseOneShots()
|
|
{
|
|
if (oneShotAudioSource != null)
|
|
{
|
|
oneShotAudioSource.UnPause();
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the length of an audio clip
|
|
/// </summary>
|
|
public static double GetClipLength(string name, float pitch = 1f, string game = null)
|
|
{
|
|
AudioClip clip = null;
|
|
if (game != null)
|
|
{
|
|
string soundName = name.Split('/')[2];
|
|
var inf = GameManager.instance.GetGameInfo(game);
|
|
//first try the game's common assetbundle
|
|
// Debug.Log("Jukebox loading sound " + soundName + " from common");
|
|
clip = inf.GetCommonAssetBundle()?.LoadAsset<AudioClip>(soundName);
|
|
//then the localized one
|
|
if (clip == null)
|
|
{
|
|
// Debug.Log("Jukebox loading sound " + soundName + " from locale");
|
|
clip = inf.GetLocalizedAssetBundle()?.LoadAsset<AudioClip>(soundName);
|
|
}
|
|
}
|
|
|
|
//can't load from assetbundle, load from resources
|
|
if (clip == null)
|
|
{
|
|
// Debug.Log("Jukebox loading sound " + name + " from resources");
|
|
clip = Resources.Load<AudioClip>($"Sfx/{name}");
|
|
}
|
|
|
|
if (clip == null)
|
|
{
|
|
Debug.LogError($"Could not load clip {name}");
|
|
return double.NaN;
|
|
}
|
|
return clip.length / pitch;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the length of an audio clip
|
|
/// Audio clip is fetched from minigame resources
|
|
/// </summary>
|
|
public static double GetClipLengthGame(string name, float pitch = 1f)
|
|
{
|
|
string gameName = name.Split('/')[0];
|
|
var inf = GameManager.instance.GetGameInfo(gameName);
|
|
if (inf != null)
|
|
{
|
|
return GetClipLength($"games/{name}", pitch, inf.usesAssetBundle ? gameName : null);
|
|
}
|
|
|
|
return double.NaN;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Fires a one-shot sound.
|
|
/// Unpitched, non-scheduled, non-looping sounds are played using a global One-Shot audio source that doesn't create a Sound object.
|
|
/// Looped sounds return their created Sound object so they can be canceled after creation.
|
|
/// </summary>
|
|
public static Sound PlayOneShot(string name, double beat = -1, float pitch = 1f, float volume = 1f, bool looping = false, string game = null, double offset = 0f)
|
|
{
|
|
AudioClip clip = null;
|
|
if (game != null)
|
|
{
|
|
string soundName = name.Split('/')[2];
|
|
var inf = GameManager.instance.GetGameInfo(game);
|
|
//first try the game's common assetbundle
|
|
// Debug.Log("Jukebox loading sound " + soundName + " from common");
|
|
clip = inf.GetCommonAssetBundle()?.LoadAsset<AudioClip>(soundName);
|
|
//then the localized one
|
|
if (clip == null)
|
|
{
|
|
// Debug.Log("Jukebox loading sound " + soundName + " from locale");
|
|
clip = inf.GetLocalizedAssetBundle()?.LoadAsset<AudioClip>(soundName);
|
|
}
|
|
}
|
|
|
|
//can't load from assetbundle, load from resources
|
|
if (clip == null)
|
|
{
|
|
// Debug.Log("Jukebox loading sound " + name + " from resources");
|
|
clip = Resources.Load<AudioClip>($"Sfx/{name}");
|
|
}
|
|
|
|
if (looping || beat != -1 || pitch != 1f)
|
|
{
|
|
Sound snd = GetAvailableScheduledSound();
|
|
|
|
snd.clip = clip;
|
|
snd.beat = beat;
|
|
snd.pitch = pitch;
|
|
snd.volume = volume;
|
|
snd.looping = looping;
|
|
snd.offset = offset;
|
|
snd.Play();
|
|
|
|
return snd;
|
|
}
|
|
else
|
|
{
|
|
if (oneShotAudioSourceObject == null)
|
|
{
|
|
oneShotAudioSourceObject = new GameObject("OneShot Audio Source");
|
|
oneShotAudioSource = oneShotAudioSourceObject.AddComponent<AudioSource>();
|
|
UnityEngine.Object.DontDestroyOnLoad(oneShotAudioSourceObject);
|
|
}
|
|
|
|
oneShotAudioSource.PlayOneShot(clip, volume);
|
|
return null;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Schedules a sound to be played at a specific time in seconds.
|
|
/// </summary>
|
|
public static Sound PlayOneShotScheduled(string name, double targetTime, float pitch = 1f, float volume = 1f, bool looping = false, string game = null)
|
|
{
|
|
Sound snd = GetAvailableScheduledSound();
|
|
|
|
AudioClip clip = null;
|
|
if (game != null)
|
|
{
|
|
string soundName = name.Split('/')[2];
|
|
var inf = GameManager.instance.GetGameInfo(game);
|
|
//first try the game's common assetbundle
|
|
// Debug.Log("Jukebox loading sound " + soundName + " from common");
|
|
clip = inf.GetCommonAssetBundle()?.LoadAsset<AudioClip>(soundName);
|
|
//then the localized one
|
|
if (clip == null)
|
|
{
|
|
// Debug.Log("Jukebox loading sound " + soundName + " from locale");
|
|
clip = inf.GetLocalizedAssetBundle()?.LoadAsset<AudioClip>(soundName);
|
|
}
|
|
}
|
|
|
|
//can't load from assetbundle, load from resources
|
|
if (clip == null)
|
|
clip = Resources.Load<AudioClip>($"Sfx/{name}");
|
|
|
|
// abort if no clip found
|
|
|
|
snd.clip = clip;
|
|
snd.pitch = pitch;
|
|
snd.volume = volume;
|
|
snd.looping = looping;
|
|
|
|
snd.scheduled = true;
|
|
snd.scheduledTime = targetTime;
|
|
snd.Play();
|
|
|
|
return snd;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Fires a one-shot sound located in minigame resources.
|
|
/// Unpitched, non-scheduled, non-looping sounds are played using a global One-Shot audio source that doesn't create a Sound object.
|
|
/// Looped sounds return their created Sound object so they can be canceled after creation.
|
|
/// </summary>
|
|
public static Sound PlayOneShotGame(string name, double beat = -1, float pitch = 1f, float volume = 1f, bool looping = false, bool forcePlay = false, double offset = 0f)
|
|
{
|
|
string gameName = name.Split('/')[0];
|
|
var inf = GameManager.instance.GetGameInfo(gameName);
|
|
if (GameManager.instance.currentGame == gameName || forcePlay)
|
|
{
|
|
return PlayOneShot($"games/{name}", beat, pitch, volume, looping, inf.usesAssetBundle ? gameName : null, offset);
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Schedules a sound to be played at a specific time in seconds.
|
|
/// Audio clip is fetched from minigame resources
|
|
/// </summary>
|
|
public static Sound PlayOneShotScheduledGame(string name, double targetTime, float pitch = 1f, float volume = 1f, bool looping = false, bool forcePlay = false)
|
|
{
|
|
string gameName = name.Split('/')[0];
|
|
var inf = GameManager.instance.GetGameInfo(gameName);
|
|
if (GameManager.instance.currentGame == gameName || forcePlay)
|
|
{
|
|
return PlayOneShotScheduled($"games/{name}", targetTime, pitch, volume, looping, inf.usesAssetBundle ? gameName : null);
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Stops a looping Sound
|
|
/// </summary>
|
|
public static void KillLoop(Sound source, float fadeTime)
|
|
{
|
|
// Safeguard against previously-destroyed sounds.
|
|
if (source == null)
|
|
return;
|
|
|
|
source.KillLoop(fadeTime);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets a pitch multiplier from semitones.
|
|
/// </summary>
|
|
public static float GetPitchFromSemiTones(int semiTones, bool pitchToMusic)
|
|
{
|
|
if (pitchToMusic)
|
|
{
|
|
return Mathf.Pow(2f, (1f / 12f) * semiTones) * Conductor.instance.musicSource.pitch;
|
|
}
|
|
else
|
|
{
|
|
return Mathf.Pow(2f, (1f / 12f) * semiTones);
|
|
}
|
|
}
|
|
/// <summary>
|
|
/// Returns the semitones from a pitch.
|
|
/// </summary>
|
|
/// <param name="pitch">The pitch of the sound.</param>
|
|
public static int GetSemitonesFromPitch(float pitch, bool pitchToMusic)
|
|
{
|
|
if (pitchToMusic) return (int)((12f * Mathf.Log(pitch, 2)) / Conductor.instance.musicSource.pitch);
|
|
return (int)(12f * Mathf.Log(pitch, 2));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets a pitch multiplier from cents.
|
|
/// </summary>
|
|
public static float GetPitchFromCents(int cents, bool pitchToMusic)
|
|
{
|
|
if (pitchToMusic)
|
|
{
|
|
return Mathf.Pow(2f, (1f / 12f) * (cents / 100)) * Conductor.instance.musicSource.pitch;
|
|
}
|
|
else
|
|
{
|
|
return Mathf.Pow(2f, (1f / 12f) * (cents / 100));
|
|
}
|
|
}
|
|
}
|
|
|
|
} |