using HeavenStudio.Util; using System; using System.Linq; using System.Collections.Generic; using UnityEngine; using Jukebox; namespace HeavenStudio.Games.Loaders { using static Minigames; public static class RvlTapTroupeLoader { public static Minigame AddGame(EventCaller eventCaller) { return new Minigame("tapTroupe", "Tap Troupe", "999999", false, false, new List() { new GameAction("stepping", "Stepping") { preFunction = delegate { var e = eventCaller.currentEntity; TapTroupe.PreStepping(e.beat, e.length, e["startTap"]); }, defaultLength = 4f, resizable = true, parameters = new List() { new Param("startTap", false, "Start Tap Voice Line", "Whether or not it should say -Tap!- on the first step.") } }, new GameAction("tapping", "Tapping") { preFunction = delegate { var e = eventCaller.currentEntity; TapTroupe.PreTapping(e.beat, e.length, e["okay"], e["okayType"], e["animType"], e["popperBeats"], e["randomVoiceLine"], e["noReady"]); }, defaultLength = 3f, resizable = true, parameters = new List() { new Param("okay", true, "Okay Voice Line", "Whether or not the tappers should say -Okay!- after successfully tapping."), new Param("okayType", TapTroupe.OkayType.OkayA, "Okay Type", "Which version of the okay voice line should the tappers say?"), new Param("animType", TapTroupe.OkayAnimType.Normal, "Okay Animation", "Which animations should be played when the tapper say OK?"), new Param("popperBeats", new EntityTypes.Float(0f, 80f, 2f), "Popper Beats", "How many beats until the popper will pop?"), new Param("randomVoiceLine", true, "Extra Random Voice Line", "Whether there should be randomly said woos or laughs after the tappers say OK!"), new Param("noReady", false, "Mute Ready") } }, new GameAction("bop", "Bop") { function = delegate {var e = eventCaller.currentEntity; TapTroupe.instance.Bop(e.beat, e.length, e["bop"], e["bopAuto"]); }, resizable = true, parameters = new List() { new Param("bop", true, "Bop", "Should the tappers bop?"), new Param("bopAuto", false, "Bop (Auto)", "Should the tappers auto bop?") } }, new GameAction("spotlights", "Toggle Spotlights") { function = delegate {var e = eventCaller.currentEntity; TapTroupe.instance.Spotlights(e["toggle"], e["player"], e["middleLeft"], e["middleRight"], e["leftMost"]); }, defaultLength = 0.5f, parameters = new List() { new Param("toggle", true, "Darkness On", "Whether or not it should be dark."), new Param("player", true, "Player Spotlight", "Whether or not the player spotlight should be turned on or off."), new Param("middleRight", false, "Middleright Tapper Spotlight", "Whether or not the middleright tapper spotlight should be turned on or off."), new Param("middleLeft", false, "Middleleft Tapper Spotlight", "Whether or not the middleleft tapper spotlight should be turned on or off."), new Param("leftMost", false, "Leftmost Tapper Spotlight", "Whether or not the leftmost tapper spotlight should be turned on or off."), } }, new GameAction("zoomOut", "Special Zoom Out") { function = delegate { TapTroupe.instance.ToggleZoomOut(); }, defaultLength = 4f, resizable = true, parameters = new List() { new Param("ease", Util.EasingFunction.Ease.EaseOutQuad, "Camera Ease", "What ease should the camera use?"), }, }, new GameAction("tutorialMissFace", "Toggle Tutorial Miss Face") { function = delegate { var e = eventCaller.currentEntity; TapTroupe.instance.ToggleMissFace(e["toggle"]); }, defaultLength = 0.5f, parameters = new List() { new Param("toggle", true, "Use it?", "Use the faces they do when you miss in the tutorial of Tap Troupe?") } } }, new List() {"rvl", "keep"}, "rvllegs", "en", new List() {"en"} ); } } } namespace HeavenStudio.Games { using Scripts_TapTroupe; public class TapTroupe : Minigame { [Header("Components")] [SerializeField] TapTroupeTapper playerTapper; [SerializeField] TapTroupeCorner playerCorner; [SerializeField] List npcTappers = new List(); [SerializeField] List npcCorners = new List(); [SerializeField] GameObject spotlightPlayer; [SerializeField] GameObject spotlightMiddleLeft; [SerializeField] GameObject spotlightMiddleRight; [SerializeField] GameObject spotlightLeftMost; [SerializeField] GameObject darkness; private Animator zoomOutAnim; [Header("Properties")] private double currentZoomCamBeat; private float currentZoomCamLength; private Util.EasingFunction.Ease lastEase; private int currentZoomIndex; private List allCameraEvents = new List(); private bool keepZoomOut; private static List queuedSteps = new List(); private static List queuedTaps = new List(); public static bool prepareTap; private bool tapping; private bool stepping; private bool shouldSwitchStep; private bool shouldDoSecondBam; private bool missedTaps; private bool canSpit = true; private bool goBop; private bool useTutorialMissFace; private TapTroupeTapper.TapAnim currentTapAnim; public GameEvent bop = new GameEvent(); public struct QueuedSteps { public double beat; public float length; public bool startTap; } public struct QueuedTaps { public double beat; public float length; public bool okay; public int okayType; public int animType; public float popperBeats; public bool randomVoiceLine; } public enum OkayType { OkayA = 0, OkayB = 1, OkayC = 2, Random = 3 } public enum OkayAnimType { Normal = 0, Popper = 1, OkSign = 2, Random = 3 } private int stepSound = 1; public static TapTroupe instance; void OnDestroy() { if (queuedSteps.Count > 0) queuedSteps.Clear(); if (queuedTaps.Count > 0) queuedTaps.Clear(); prepareTap = false; foreach (var evt in scheduledInputs) { evt.Disable(); } } public override void OnTimeChange() { UpdateCameraZoom(); } void Awake() { instance = this; zoomOutAnim = GetComponent(); var camEvents = EventCaller.GetAllInGameManagerList("tapTroupe", new string[] { "zoomOut" }); List tempEvents = new List(); for (int i = 0; i < camEvents.Count; i++) { if (camEvents[i].beat + camEvents[i].beat >= Conductor.instance.songPositionInBeatsAsDouble) { tempEvents.Add(camEvents[i]); } } allCameraEvents = tempEvents; UpdateCameraZoom(); } void Update() { var cond = Conductor.instance; if (cond.isPlaying && !cond.isPaused) { if (cond.ReportBeat(ref bop.lastReportedBeat, bop.startBeat % 1) && goBop) { BopSingle(); } if (queuedSteps.Count > 0) { foreach (var step in queuedSteps) { Stepping(step.beat, step.length, step.startTap); } queuedSteps.Clear(); } if (queuedTaps.Count > 0) { foreach (var tap in queuedTaps) { Tapping(tap.beat, tap.length, tap.okay, tap.okayType, tap.animType, tap.popperBeats, tap.randomVoiceLine); BeatAction.New(instance.gameObject, new List() { new BeatAction.Action(tap.beat - 1.1f, delegate { prepareTap = true; }), new BeatAction.Action(tap.beat, delegate { prepareTap = false; }) }); } queuedTaps.Clear(); } if (PlayerInput.Pressed() && !IsExpectingInputNow(InputType.STANDARD_DOWN)) { if (canSpit && !useTutorialMissFace) SoundByte.PlayOneShotGame("tapTroupe/spit", -1, 1, 0.5f); SoundByte.PlayOneShotGame("tapTroupe/miss"); TapTroupe.instance.ScoreMiss(0.5f); foreach (var corner in npcCorners) { if (useTutorialMissFace) { corner.SetMissFace(TapTroupeCorner.MissFace.LOL); } else { corner.SetMissFace(TapTroupeCorner.MissFace.Spit); } } if (tapping) { missedTaps = true; playerTapper.Tap(currentTapAnim, false, shouldSwitchStep); playerCorner.Bop(); } else { playerTapper.Step(false); playerCorner.Bop(); } canSpit = false; } } if (allCameraEvents.Count > 0) { if (currentZoomIndex < allCameraEvents.Count && currentZoomIndex >= 0) { if (Conductor.instance.songPositionInBeatsAsDouble >= allCameraEvents[currentZoomIndex].beat) { UpdateCameraZoom(); currentZoomIndex++; } } float normalizedBeat = Conductor.instance.GetPositionFromBeat(currentZoomCamBeat, currentZoomCamLength); float normalizedAnimBeat = normalizedBeat * 4; if (normalizedBeat >= 0) { if (!keepZoomOut) { GameCamera.additionalPosition = new Vector3(0, 0, 0); zoomOutAnim.Play("NoZoomOut", 0, 0); } else { Util.EasingFunction.Function func = Util.EasingFunction.GetEasingFunction(lastEase); if (normalizedBeat > 1) GameCamera.additionalPosition = new Vector3(0, 30, -100); else { float newPosY = func(0, 30, normalizedBeat); float newPosZ = func(0, -100, normalizedBeat); GameCamera.additionalPosition = new Vector3(0, newPosY, newPosZ); } if (normalizedAnimBeat > 1) { zoomOutAnim.DoNormalizedAnimation("ZoomOut", 1f); playerTapper.FadeOut(1f); foreach (var tapper in npcTappers) { tapper.FadeOut(1f); } } else { zoomOutAnim.DoNormalizedAnimation("ZoomOut", normalizedAnimBeat); playerTapper.FadeOut(normalizedAnimBeat); foreach (var tapper in npcTappers) { tapper.FadeOut(normalizedAnimBeat); } } } } } } void UpdateCameraZoom() { if (currentZoomIndex < allCameraEvents.Count && currentZoomIndex >= 0) { currentZoomCamLength = allCameraEvents[currentZoomIndex].length; currentZoomCamBeat = allCameraEvents[currentZoomIndex].beat; lastEase = (Util.EasingFunction.Ease)allCameraEvents[currentZoomIndex]["ease"]; } } public void ToggleZoomOut() { keepZoomOut = true; } public static void PreStepping(double beat, float length, bool startTap) { if (GameManager.instance.currentGame == "tapTroupe") { TapTroupe.instance.Stepping(beat, length, startTap); } else { queuedSteps.Add(new QueuedSteps { beat = beat, length = length, startTap = startTap }); } } public void Stepping(double beat, float length, bool startTap) { for (int i = 0; i < length; i++) { TapTroupe.instance.ScheduleInput(beat - 1, 1 + i, InputType.STANDARD_DOWN, TapTroupe.instance.JustStep, TapTroupe.instance.MissStep, TapTroupe.instance.Nothing); BeatAction.New(instance.gameObject, new List() { new BeatAction.Action(beat + i, delegate { TapTroupe.instance.NPCStep(); SoundByte.PlayOneShotGame("tapTroupe/other1", -1, 1, 0.75f); }) }); } BeatAction.New(instance.gameObject, new List() { new BeatAction.Action(beat - 1, delegate { if (tapping) return; TapTroupe.instance.NPCStep(false, false); TapTroupe.instance.playerTapper.Step(false, false); TapTroupe.instance.playerCorner.Bop(); }), new BeatAction.Action(beat, delegate { if (startTap) SoundByte.PlayOneShotGame("tapTroupe/startTap"); stepping = true; }), new BeatAction.Action(beat + length + 1, delegate { stepping = false; }), }); } public static void PreTapping(double beat, float length, bool okay, int okayType, int animType, float popperBeats, bool randomVoiceLine, bool noReady) { if (!noReady) { MultiSound.Play(new MultiSound.Sound[] { new MultiSound.Sound("tapTroupe/tapReady1", beat - 2f), new MultiSound.Sound("tapTroupe/tapReady2", beat - 1f), }, forcePlay: true); } if (GameManager.instance.currentGame == "tapTroupe") { TapTroupe.instance.Tapping(beat, length, okay, okayType, animType, popperBeats, randomVoiceLine); BeatAction.New(instance.gameObject, new List() { new BeatAction.Action(beat - 1.1f, delegate { prepareTap = true; }), new BeatAction.Action(beat, delegate { prepareTap = false; }) }); } else { queuedTaps.Add(new QueuedTaps { beat = beat, length = length, okay = okay, okayType = okayType, animType = animType, popperBeats = popperBeats, randomVoiceLine = randomVoiceLine }); } } public void Tapping(double beat, float length, bool okay, int okayType, int animType, float popperBeats, bool randomVoiceLine) { float actualLength = length - 0.5f; actualLength -= actualLength % 0.75f; bool secondBam = false; double finalBeatToSpawn = 0f; if (actualLength < 2.25f) actualLength = 2.25f; List soundsToPlay = new List { new MultiSound.Sound("tapTroupe/tapAnd", beat) }; for (float i = 0; i < actualLength; i += 0.75f) { string soundToPlay = "bamvoice1"; string otherSoundToPlay = "other3"; double beatToSpawn = beat + i + 0.5f; if (i + 0.75f >= actualLength) { soundToPlay = "startTap"; otherSoundToPlay = "other2"; beatToSpawn = Math.Ceiling(beat + i); finalBeatToSpawn = beatToSpawn; BeatAction.New(instance.gameObject, new List() { new BeatAction.Action(beatToSpawn - 0.3f, delegate { currentTapAnim = TapTroupeTapper.TapAnim.LastTap; shouldSwitchStep = false; }), new BeatAction.Action(beatToSpawn, delegate { NPCTap(TapTroupeTapper.TapAnim.LastTap, true, false);}), new BeatAction.Action(beatToSpawn + 0.1f, delegate { tapping = false; }) }); } else if (i + 1.5f >= actualLength) { soundToPlay = "tapvoice2"; BeatAction.New(instance.gameObject, new List() { new BeatAction.Action(beatToSpawn - 0.3f, delegate { currentTapAnim = TapTroupeTapper.TapAnim.Tap; shouldSwitchStep = false; }), new BeatAction.Action(beatToSpawn, delegate { NPCTap(TapTroupeTapper.TapAnim.Tap, true, false); }) }); } else if (i + 2.25f >= actualLength) { soundToPlay = "tapvoice1"; if (actualLength == 2.25f) { BeatAction.New(instance.gameObject, new List() { new BeatAction.Action(beatToSpawn - 0.3f, delegate { currentTapAnim = TapTroupeTapper.TapAnim.Tap; shouldSwitchStep = true; }), new BeatAction.Action(beatToSpawn, delegate { NPCTap(TapTroupeTapper.TapAnim.Tap); }) }); } else { BeatAction.New(instance.gameObject, new List() { new BeatAction.Action(beatToSpawn - 0.3f, delegate { currentTapAnim = TapTroupeTapper.TapAnim.Tap; shouldSwitchStep = false; }), new BeatAction.Action(beatToSpawn, delegate { NPCTap(TapTroupeTapper.TapAnim.Tap, true, false); }) }); } } else { if (secondBam) soundToPlay = "bamvoice2"; if (i + 3f >= actualLength) { if (actualLength == 3f) { BeatAction.New(instance.gameObject, new List() { new BeatAction.Action(beatToSpawn - 0.3f, delegate { currentTapAnim = TapTroupeTapper.TapAnim.Tap; shouldSwitchStep = true; }), new BeatAction.Action(beatToSpawn, delegate { NPCTap(TapTroupeTapper.TapAnim.Tap); }) }); } else { BeatAction.New(instance.gameObject, new List() { new BeatAction.Action(beatToSpawn - 0.3f, delegate { currentTapAnim = TapTroupeTapper.TapAnim.BamTapReady; shouldSwitchStep = true; }), new BeatAction.Action(beatToSpawn, delegate { NPCTap(TapTroupeTapper.TapAnim.BamTapReady); }) }); } } else if (i == 0) { BeatAction.New(instance.gameObject, new List() { new BeatAction.Action(beatToSpawn - 0.3f, delegate { currentTapAnim = TapTroupeTapper.TapAnim.BamReady; shouldSwitchStep = false; }), new BeatAction.Action(beatToSpawn, delegate { NPCTap(TapTroupeTapper.TapAnim.BamReady, true, false); }) }); } else { BeatAction.New(instance.gameObject, new List() { new BeatAction.Action(beatToSpawn - 0.3f, delegate { currentTapAnim = TapTroupeTapper.TapAnim.Bam; shouldSwitchStep = true; }), new BeatAction.Action(beatToSpawn, delegate { NPCTap(TapTroupeTapper.TapAnim.Bam); }), new BeatAction.Action(beatToSpawn + 0.1f, delegate { if (playerTapper.transform.localScale.x != npcTappers[0].transform.localScale.x) playerTapper.dontSwitchNextStep = true; }) }); } } soundsToPlay.Add(new MultiSound.Sound($"tapTroupe/{soundToPlay}", beatToSpawn)); soundsToPlay.Add(new MultiSound.Sound($"tapTroupe/{otherSoundToPlay}", beatToSpawn)); shouldDoSecondBam = secondBam; secondBam = !secondBam; ScheduleInput(beatToSpawn - 1, 1f, InputType.STANDARD_DOWN, JustTap, MissTap, Nothing); } int actualOkayType = okayType; if (actualOkayType == (int)OkayType.Random) actualOkayType = UnityEngine.Random.Range(0, 3); int randomO = UnityEngine.Random.Range(0, 3); string okayVoiceLine = "A"; string okayOneVoiceLine = "A"; switch (actualOkayType) { case (int)OkayType.OkayA: okayVoiceLine = "A"; break; case (int)OkayType.OkayB: okayVoiceLine = "B"; break; case (int)OkayType.OkayC: okayVoiceLine = "C"; break; default: okayVoiceLine = "A"; break; } switch (randomO) { case (int)OkayType.OkayA: okayOneVoiceLine = "A"; break; case (int)OkayType.OkayB: okayOneVoiceLine = "B"; break; case (int)OkayType.OkayC: okayOneVoiceLine = "C"; break; default: okayOneVoiceLine = "A"; break; } BeatAction.New(instance.gameObject, new List() { new BeatAction.Action(beat - 1f, delegate { if (!stepping) { NPCStep(false); playerTapper.Step(false); playerCorner.Bop(); } }), new BeatAction.Action(beat, delegate { tapping = true; missedTaps = false; }), new BeatAction.Action(finalBeatToSpawn, delegate { if (missedTaps || animType != (int)OkayAnimType.Popper) return; npcCorners[0].PartyPopper(finalBeatToSpawn + popperBeats); }), new BeatAction.Action(finalBeatToSpawn + 0.5f, delegate { if (missedTaps || !okay) return; playerCorner.Okay(); foreach (var corner in npcCorners) { corner.Okay(); } MultiSound.Play(new MultiSound.Sound[] { new MultiSound.Sound($"tapTroupe/okay{okayOneVoiceLine}1", finalBeatToSpawn + 0.5f), new MultiSound.Sound($"tapTroupe/okay{okayVoiceLine}2", finalBeatToSpawn + 1f, 1f, 0.75f), }, forcePlay: true); }), new BeatAction.Action(finalBeatToSpawn + 1f, delegate { if (!missedTaps && okay && randomVoiceLine && UnityEngine.Random.Range(1, 50) == 1) { SoundByte.PlayOneShotGame("tapTroupe/woo"); } else if (!missedTaps && okay && randomVoiceLine && UnityEngine.Random.Range(1, 50) == 1) { SoundByte.PlayOneShotGame("tapTroupe/laughter", -1, 1, 0.4f); } if (missedTaps || animType != (int)OkayAnimType.OkSign) return; playerCorner.OkaySign(); foreach (var corner in npcCorners) { corner.OkaySign(); } }) }); MultiSound.Play(soundsToPlay.ToArray(), forcePlay: true); } public void Bop(double beat, float length, bool shouldBop, bool autoBop) { goBop = autoBop; if (shouldBop) { for (int i = 0; i < length; i++) { BeatAction.New(instance.gameObject, new List() { new BeatAction.Action(beat + i, delegate { BopSingle(); }) }); } } } public void BopSingle() { playerTapper.Bop(); playerCorner.Bop(); foreach (var tapper in npcTappers) { tapper.Bop(); } foreach (var corner in npcCorners) { corner.Bop(); } } public void ToggleMissFace(bool isOn) { useTutorialMissFace = isOn; } public void Spotlights(bool isOn, bool player, bool middleLeft, bool middleRight, bool leftMost) { if (isOn) { darkness.SetActive(true); spotlightPlayer.SetActive(player); spotlightMiddleLeft.SetActive(middleLeft); spotlightMiddleRight.SetActive(middleRight); spotlightLeftMost.SetActive(leftMost); } else { darkness.SetActive(false); spotlightPlayer.SetActive(false); spotlightMiddleLeft.SetActive(false); spotlightMiddleRight.SetActive(false); spotlightLeftMost.SetActive(false); } } public void NPCStep(bool hit = true, bool switchFeet = true) { foreach (var tapper in npcTappers) { tapper.Step(hit, switchFeet); } foreach (var corner in npcCorners) { corner.Bop(); } } public void NPCTap(TapTroupeTapper.TapAnim animType, bool hit = true, bool switchFeet = true) { foreach (var tapper in npcTappers) { tapper.Tap(animType, hit, switchFeet); } foreach (var corner in npcCorners) { corner.Bop(); } } void JustStep(PlayerActionEvent caller, float state) { if (state >= 1f || state <= -1f) { canSpit = true; playerTapper.Step(false); playerCorner.Bop(); SoundByte.PlayOneShotGame("tapTroupe/tink"); if (stepSound == 1) { stepSound = 2; } else { stepSound = 1; } foreach (var corner in npcCorners) { if (useTutorialMissFace) { corner.SetMissFace(TapTroupeCorner.MissFace.LOL); } else { corner.SetMissFace(TapTroupeCorner.MissFace.Sad); } } return; } SuccessStep(caller); } void SuccessStep(PlayerActionEvent caller) { canSpit = true; playerTapper.Step(); playerCorner.Bop(); SoundByte.PlayOneShotGame($"tapTroupe/step{stepSound}"); if (stepSound == 1) { stepSound = 2; } else { stepSound = 1; } foreach (var corner in npcCorners) { corner.ResetFace(); } BeatAction.New(instance.gameObject, new List() { new BeatAction.Action(caller.startBeat + caller.timer + 0.1f, delegate { if (playerTapper.transform.localScale.x != npcTappers[0].transform.localScale.x) playerTapper.dontSwitchNextStep = true; }) }); } void MissStep(PlayerActionEvent caller) { if (canSpit && !useTutorialMissFace) SoundByte.PlayOneShotGame("tapTroupe/spit", -1, 1, 0.5f); foreach (var corner in npcCorners) { if (useTutorialMissFace) { corner.SetMissFace(TapTroupeCorner.MissFace.LOL); } else { corner.SetMissFace(TapTroupeCorner.MissFace.Spit); } } canSpit = false; } void JustTap(PlayerActionEvent caller, float state) { if (state >= 1f || state <= -1f) { missedTaps = true; canSpit = true; playerTapper.Tap(currentTapAnim, false, shouldSwitchStep); playerCorner.Bop(); switch (currentTapAnim) { case TapTroupeTapper.TapAnim.LastTap: SoundByte.PlayOneShotGame("tapTroupe/tap3"); break; default: SoundByte.PlayOneShotGame("tapTroupe/tink"); break; } foreach (var corner in npcCorners) { if (useTutorialMissFace) { corner.SetMissFace(TapTroupeCorner.MissFace.LOL); } else { corner.SetMissFace(TapTroupeCorner.MissFace.Sad); } } return; } SuccessTap(); } void SuccessTap() { canSpit = true; playerTapper.Tap(currentTapAnim, true, shouldSwitchStep); playerCorner.Bop(); switch (currentTapAnim) { case TapTroupeTapper.TapAnim.LastTap: SoundByte.PlayOneShotGame("tapTroupe/tap3"); break; default: SoundByte.PlayOneShotGame("tapTroupe/player3"); break; } foreach (var corner in npcCorners) { corner.ResetFace(); } } void MissTap(PlayerActionEvent caller) { missedTaps = true; if (canSpit && !useTutorialMissFace) SoundByte.PlayOneShotGame("tapTroupe/spit", -1, 1, 0.5f); foreach (var corner in npcCorners) { if (useTutorialMissFace) { corner.SetMissFace(TapTroupeCorner.MissFace.LOL); } else { corner.SetMissFace(TapTroupeCorner.MissFace.Spit); } } canSpit = false; } void Nothing(PlayerActionEvent caller) { } } }