From 014a8f320b5aba2957ceec88e36834d9e2c341ba Mon Sep 17 00:00:00 2001
From: Cameron Taylor <cameron.taylor.ninja@gmail.com>
Date: Wed, 15 Mar 2023 21:05:15 -0400
Subject: [PATCH 1/5] freeplay exiting stuff

---
 source/funkin/FreeplayState.hx              | 213 ++++++++++++++------
 source/funkin/freeplayStuff/SongMenuItem.hx |  40 +++-
 2 files changed, 182 insertions(+), 71 deletions(-)

diff --git a/source/funkin/FreeplayState.hx b/source/funkin/FreeplayState.hx
index 34794b6ed..20dca2eb7 100644
--- a/source/funkin/FreeplayState.hx
+++ b/source/funkin/FreeplayState.hx
@@ -70,9 +70,8 @@ class FreeplayState extends MusicBeatSubstate
 
   var dj:DJBoyfriend;
 
-  var iconArray:Array<HealthIcon> = [];
-
   var typing:FlxInputText;
+  var exitMovers:Map<FlxSprite, MoveData> = new Map();
 
   override function create()
   {
@@ -133,6 +132,14 @@ class FreeplayState extends MusicBeatSubstate
     pinkBack.color = 0xFFffd4e9; // sets it to pink!
     pinkBack.x -= pinkBack.width;
 
+    exitMovers.set(pinkBack,
+      {
+        x: -pinkBack.width,
+        y: pinkBack.y,
+        speed: 0.4,
+        wait: 0
+      });
+
     FlxTween.tween(pinkBack, {x: 0}, 0.6, {ease: FlxEase.quartOut});
     add(pinkBack);
 
@@ -155,30 +162,84 @@ class FreeplayState extends MusicBeatSubstate
     moreWays.speed = 4;
     grpTxtScrolls.add(moreWays);
 
+    exitMovers.set(moreWays,
+      {
+        x: FlxG.width,
+        y: moreWays.y,
+        speed: 0.3,
+        wait: 0
+      });
+
     var funnyScroll:BGScrollingText = new BGScrollingText(0, 250, "BOYFRIEND", FlxG.width / 2);
     funnyScroll.funnyColor = 0xFFff9963;
     funnyScroll.speed = -1;
     grpTxtScrolls.add(funnyScroll);
 
+    exitMovers.set(funnyScroll,
+      {
+        x: -funnyScroll.width,
+        y: funnyScroll.y,
+        speed: 0.3,
+        wait: 0
+      });
+
     var txtNuts:BGScrollingText = new BGScrollingText(0, 300, "PROTECT YO NUTS", FlxG.width / 2);
     grpTxtScrolls.add(txtNuts);
+    exitMovers.set(txtNuts,
+      {
+        x: FlxG.width,
+        y: txtNuts.y,
+        speed: 0.3,
+        wait: 0
+      });
 
     var funnyScroll2:BGScrollingText = new BGScrollingText(0, 340, "BOYFRIEND", FlxG.width / 2);
     funnyScroll2.funnyColor = 0xFFff9963;
     funnyScroll2.speed = -1.2;
     grpTxtScrolls.add(funnyScroll2);
 
+    exitMovers.set(funnyScroll2,
+      {
+        x: -funnyScroll2.width,
+        y: funnyScroll2.y,
+        speed: 0.3,
+        wait: 0
+      });
+
     var moreWays2:BGScrollingText = new BGScrollingText(0, 400, "HOT BLOODED IN MORE WAYS THAN ONE", FlxG.width);
     moreWays2.funnyColor = 0xFFfff383;
     moreWays2.speed = 4.4;
     grpTxtScrolls.add(moreWays2);
 
+    exitMovers.set(moreWays2,
+      {
+        x: FlxG.width,
+        y: moreWays2.y,
+        speed: 0.3,
+        wait: 0
+      });
+
     var funnyScroll3:BGScrollingText = new BGScrollingText(0, orangeBackShit.y, "BOYFRIEND", FlxG.width / 2);
     funnyScroll3.funnyColor = 0xFFff9963;
     funnyScroll3.speed = -0.8;
     grpTxtScrolls.add(funnyScroll3);
 
+    exitMovers.set(funnyScroll3,
+      {
+        x: -funnyScroll3.width,
+        y: funnyScroll3.y,
+        speed: 0.3,
+        wait: 0
+      });
+
     dj = new DJBoyfriend(0, -100);
+    exitMovers.set(dj,
+      {
+        x: -dj.width * 1.6,
+        y: dj.y,
+        speed: 0.5,
+        wait: 0
+      });
     add(dj);
 
     var bgDad:FlxSprite = new FlxSprite(pinkBack.width * 0.75, 0).loadGraphic(Paths.image('freeplay/freeplayBGdad'));
@@ -187,9 +248,25 @@ class FreeplayState extends MusicBeatSubstate
     bgDad.shader = new AngleMask();
     bgDad.visible = false;
 
+    exitMovers.set(bgDad,
+      {
+        x: FlxG.width * 1.5,
+        y: 0,
+        speed: 0.5,
+        wait: 0
+      });
+
     var blackOverlayBullshitLOLXD:FlxSprite = new FlxSprite(FlxG.width).makeGraphic(Std.int(bgDad.width), Std.int(bgDad.height), FlxColor.BLACK);
     add(blackOverlayBullshitLOLXD); // used to mask the text lol!
 
+    exitMovers.set(blackOverlayBullshitLOLXD,
+      {
+        x: FlxG.width * 1.5,
+        y: bgDad.height,
+        speed: 0.5,
+        wait: 0
+      });
+
     add(bgDad);
     FlxTween.tween(blackOverlayBullshitLOLXD, {x: pinkBack.width * 0.75}, 1, {ease: FlxEase.quintOut});
 
@@ -208,8 +285,7 @@ class FreeplayState extends MusicBeatSubstate
     grpDifficulties.add(new FlxSprite().loadGraphic(Paths.image('freeplay/freeplayNorm')));
     grpDifficulties.add(new FlxSprite().loadGraphic(Paths.image('freeplay/freeplayHard')));
 
-    grpDifficulties.group.forEach(function(spr)
-    {
+    grpDifficulties.group.forEach(function(spr) {
       spr.visible = false;
     });
 
@@ -220,9 +296,26 @@ class FreeplayState extends MusicBeatSubstate
     add(overhangStuff);
     FlxTween.tween(overhangStuff, {y: 0}, 0.3, {ease: FlxEase.quartOut});
 
+    exitMovers.set(overhangStuff,
+      {
+        y: -overhangStuff.height,
+        x: 0,
+        speed: 0.2,
+        wait: 0
+      });
+
     var fnfFreeplay:FlxText = new FlxText(0, 12, 0, "FREEPLAY", 48);
     fnfFreeplay.font = "VCR OSD Mono";
     fnfFreeplay.visible = false;
+
+    exitMovers.set(fnfFreeplay,
+      {
+        y: fnfFreeplay.y - 64,
+        x: 0,
+        speed: 0.2,
+        wait: 0
+      });
+
     var sillyStroke = new StrokeShader(0xFFFFFFFF, 2, 2);
     fnfFreeplay.shader = sillyStroke;
     add(fnfFreeplay);
@@ -236,8 +329,7 @@ class FreeplayState extends MusicBeatSubstate
     fnfHighscoreSpr.updateHitbox();
     add(fnfHighscoreSpr);
 
-    new FlxTimer().start(FlxG.random.float(12, 50), function(tmr)
-    {
+    new FlxTimer().start(FlxG.random.float(12, 50), function(tmr) {
       fnfHighscoreSpr.animation.play("highscore");
       tmr.time = FlxG.random.float(20, 60);
     }, 0);
@@ -251,8 +343,7 @@ class FreeplayState extends MusicBeatSubstate
     txtCompletion.visible = false;
     add(txtCompletion);
 
-    dj.onIntroDone.add(function()
-    {
+    dj.onIntroDone.add(function() {
       FlxTween.tween(grpDifficulties, {x: 90}, 0.6, {ease: FlxEase.quartOut});
 
       add(new DifficultySelector(20, grpDifficulties.y - 10, false, controls));
@@ -261,8 +352,7 @@ class FreeplayState extends MusicBeatSubstate
       var letterSort:LetterSort = new LetterSort(300, 100);
       add(letterSort);
 
-      letterSort.changeSelectionCallback = (str) ->
-      {
+      letterSort.changeSelectionCallback = (str) -> {
         switch (str)
         {
           case "fav":
@@ -274,8 +364,7 @@ class FreeplayState extends MusicBeatSubstate
         }
       };
 
-      new FlxTimer().start(1 / 24, function(handShit)
-      {
+      new FlxTimer().start(1 / 24, function(handShit) {
         fnfHighscoreSpr.visible = true;
         fnfFreeplay.visible = true;
         fp.visible = true;
@@ -284,8 +373,7 @@ class FreeplayState extends MusicBeatSubstate
         txtCompletion.visible = true;
         intendedCompletion = 0;
 
-        new FlxTimer().start(1.5 / 24, function(bold)
-        {
+        new FlxTimer().start(1.5 / 24, function(bold) {
           sillyStroke.width = 0;
           sillyStroke.height = 0;
         });
@@ -333,14 +421,12 @@ class FreeplayState extends MusicBeatSubstate
     typing = new FlxInputText(100, 100);
     add(typing);
 
-    typing.callback = function(txt, action)
-    {
+    typing.callback = function(txt, action) {
       // generateSongList(new EReg(txt.trim(), "ig"));
       trace(action);
     };
 
-    forEach(function(bs)
-    {
+    forEach(function(bs) {
       bs.cameras = [funnyCam];
     });
 
@@ -361,15 +447,13 @@ class FreeplayState extends MusicBeatSubstate
       switch (filterStuff.filterType)
       {
         case STARTSWITH:
-          tempSongs = tempSongs.filter(str ->
-          {
+          tempSongs = tempSongs.filter(str -> {
             return str.songName.toLowerCase().startsWith(filterStuff.filterData);
           });
         case ALL:
         // no filter!
         case FAVORITE:
-          tempSongs = tempSongs.filter(str ->
-          {
+          tempSongs = tempSongs.filter(str -> {
             return str.isFav;
           });
         default:
@@ -406,20 +490,17 @@ class FreeplayState extends MusicBeatSubstate
 
       var maxTimer:Float = Math.min(i, 4);
 
-      new FlxTimer().start((1 / 24) * maxTimer, function(doShit)
-      {
+      new FlxTimer().start((1 / 24) * maxTimer, function(doShit) {
         funnyMenu.doJumpIn = true;
       });
 
-      new FlxTimer().start((0.09 * maxTimer) + 0.85, function(lerpTmr)
-      {
+      new FlxTimer().start((0.09 * maxTimer) + 0.85, function(lerpTmr) {
         funnyMenu.doLerp = true;
       });
 
       if (!force)
       {
-        new FlxTimer().start(((0.20 * maxTimer) / (1 + maxTimer)) + 0.75, function(swagShi)
-        {
+        new FlxTimer().start(((0.20 * maxTimer) / (1 + maxTimer)) + 0.75, function(swagShi) {
           funnyMenu.songText.visible = true;
           funnyMenu.alpha = 1;
         });
@@ -439,13 +520,6 @@ class FreeplayState extends MusicBeatSubstate
 
       // grpSongs.add(songText);
 
-      var icon:HealthIcon = new HealthIcon(tempSongs[i].songCharacter);
-      // icon.sprTracker = songText;
-
-      // using a FlxGroup is too much fuss!
-      iconArray.push(icon);
-      // add(icon);
-
       // songText.x += 40;
       // DONT PUT X IN THE FIRST PARAMETER OF new ALPHABET() !!
       // songText.screenCenter(X);
@@ -500,8 +574,7 @@ class FreeplayState extends MusicBeatSubstate
         FlxTween.tween(grpCapsules.members[realShit], {angle: 360}, 0.4,
           {
             ease: FlxEase.elasticOut,
-            onComplete: _ ->
-            {
+            onComplete: _ -> {
               grpCapsules.members[realShit].favIcon.visible = true;
               grpCapsules.members[realShit].favIcon.animation.play("fav");
             }
@@ -510,12 +583,10 @@ class FreeplayState extends MusicBeatSubstate
       else
       {
         grpCapsules.members[realShit].favIcon.animation.play('fav', false, true);
-        new FlxTimer().start((1 / 24) * 14, _ ->
-        {
+        new FlxTimer().start((1 / 24) * 14, _ -> {
           grpCapsules.members[realShit].favIcon.visible = false;
         });
-        new FlxTimer().start((1 / 24) * 24, _ ->
-        {
+        new FlxTimer().start((1 / 24) * 24, _ -> {
           FlxTween.tween(grpCapsules.members[realShit], {angle: 0}, 0.4, {ease: FlxEase.elasticOut});
         });
       }
@@ -537,7 +608,7 @@ class FreeplayState extends MusicBeatSubstate
     fp.updateScore(Std.int(lerpScore));
 
     txtCompletion.text = Math.floor(lerpCompletion * 100) + "%";
-    trace(Highscore.getCompletion(songs[curSelected].songName, curDifficulty));
+    // trace(Highscore.getCompletion(songs[curSelected].songName, curDifficulty));
 
     // trace(intendedScore);
     // trace(lerpScore);
@@ -565,7 +636,7 @@ class FreeplayState extends MusicBeatSubstate
 
           FlxG.watch.addQuick("LENGTH", length);
           FlxG.watch.addQuick("ANGLE", Math.round(FlxAngle.asDegrees(angle)));
-          trace("ANGLE", Math.round(FlxAngle.asDegrees(angle)));
+          // trace("ANGLE", Math.round(FlxAngle.asDegrees(angle)));
         }
 
         /* switch (inputID)
@@ -684,9 +755,30 @@ class FreeplayState extends MusicBeatSubstate
     {
       FlxG.sound.play(Paths.sound('cancelMenu'));
 
-      FlxTransitionableState.skipNextTransIn = true;
-      FlxTransitionableState.skipNextTransOut = true;
-      FlxG.switchState(new MainMenuState());
+      // FlxTween.tween(dj, {x: -dj.width}, 0.2, {ease: FlxEase.quartOut});
+
+      var longestTimer:Float = 0;
+
+      for (spr in exitMovers.keys())
+      {
+        var moveData:MoveData = exitMovers.get(spr);
+        FlxTween.tween(spr, {x: moveData.x, y: moveData.y}, moveData.speed, {ease: FlxEase.expoIn});
+
+        longestTimer = Math.max(longestTimer, moveData.speed + moveData.wait);
+      }
+
+      for (caps in grpCapsules.members)
+      {
+        caps.doJumpIn = false;
+        caps.doLerp = false;
+        caps.doJumpOut = true;
+      }
+
+      new FlxTimer().start(longestTimer, (_) -> {
+        FlxTransitionableState.skipNextTransIn = true;
+        FlxTransitionableState.skipNextTransOut = true;
+        FlxG.switchState(new MainMenuState());
+      });
     }
 
     if (accepted)
@@ -724,14 +816,13 @@ class FreeplayState extends MusicBeatSubstate
       SongLoad.curDiff = PlayState.storyDifficulty_NEW;
 
       PlayState.storyWeek = songs[curSelected].week;
-      trace(' CUR WEEK ' + PlayState.storyWeek);
+      // trace(' CUR WEEK ' + PlayState.storyWeek);
 
       // Visual and audio effects.
       FlxG.sound.play(Paths.sound('confirmMenu'));
       dj.confirm();
 
-      new FlxTimer().start(1, function(tmr:FlxTimer)
-      {
+      new FlxTimer().start(1, function(tmr:FlxTimer) {
         LoadingState.loadAndSwitchState(new PlayState(), true);
       });
     }
@@ -769,8 +860,7 @@ class FreeplayState extends MusicBeatSubstate
         'normal';
     };
 
-    grpDifficulties.group.forEach(function(spr)
-    {
+    grpDifficulties.group.forEach(function(spr) {
       spr.visible = false;
     });
 
@@ -779,8 +869,7 @@ class FreeplayState extends MusicBeatSubstate
     curShit.visible = true;
     curShit.offset.y += 5;
     curShit.alpha = 0.5;
-    new FlxTimer().start(1 / 24, function(swag)
-    {
+    new FlxTimer().start(1 / 24, function(swag) {
       curShit.alpha = 1;
       curShit.updateHitbox();
     });
@@ -826,13 +915,6 @@ class FreeplayState extends MusicBeatSubstate
 
     var bullShit:Int = 0;
 
-    for (i in 0...iconArray.length)
-    {
-      iconArray[i].alpha = 0.6;
-    }
-
-    iconArray[curSelected].alpha = 1;
-
     for (index => capsule in grpCapsules.members)
     {
       capsule.selected = false;
@@ -884,8 +966,7 @@ class DifficultySelector extends FlxSprite
 
     whiteShader.colorSet = true;
 
-    new FlxTimer().start(2 / 24, function(tmr)
-    {
+    new FlxTimer().start(2 / 24, function(tmr) {
       whiteShader.colorSet = false;
       updateHitbox();
     });
@@ -920,3 +1001,11 @@ class SongMetadata
     this.isFav = isFav;
   }
 }
+
+typedef MoveData =
+{
+  var x:Float;
+  var y:Float;
+  var speed:Float;
+  var wait:Float;
+}
diff --git a/source/funkin/freeplayStuff/SongMenuItem.hx b/source/funkin/freeplayStuff/SongMenuItem.hx
index 6ed3ae57d..a32b387a3 100644
--- a/source/funkin/freeplayStuff/SongMenuItem.hx
+++ b/source/funkin/freeplayStuff/SongMenuItem.hx
@@ -23,6 +23,8 @@ class SongMenuItem extends FlxSpriteGroup
   public var doLerp:Bool = false;
   public var doJumpIn:Bool = false;
 
+  public var doJumpOut:Bool = false;
+
   public function new(x:Float, y:Float, song:String)
   {
     super(x, y);
@@ -52,27 +54,47 @@ class SongMenuItem extends FlxSpriteGroup
     selected = selected; // just to kickstart the set_selected
   }
 
-  var frameTicker:Float = 0;
-  var frameTypeBeat:Int = 0;
+  var frameInTicker:Float = 0;
+  var frameInTypeBeat:Int = 0;
+
+  var frameOutTicker:Float = 0;
+  var frameOutTypeBeat:Int = 0;
 
   var xFrames:Array<Float> = [1.7, 1.8, 0.85, 0.85, 0.97, 0.97, 1];
   var xPosLerpLol:Array<Float> = [0.9, 0.4, 0.16, 0.16, 0.22, 0.22, 0.245]; // NUMBERS ARE JANK CUZ THE SCALING OR WHATEVER
+  var xPosOutLerpLol:Array<Float> = [0.245, 0.75, 0.98, 0.98, 1.2]; // NUMBERS ARE JANK CUZ THE SCALING OR WHATEVER
 
   override function update(elapsed:Float)
   {
     if (doJumpIn)
     {
-      frameTicker += elapsed;
+      frameInTicker += elapsed;
 
-      if (frameTicker >= 1 / 24 && frameTypeBeat < xFrames.length)
+      if (frameInTicker >= 1 / 24 && frameInTypeBeat < xFrames.length)
       {
-        frameTicker = 0;
+        frameInTicker = 0;
 
-        scale.x = xFrames[frameTypeBeat];
-        scale.y = 1 / xFrames[frameTypeBeat];
-        x = FlxG.width * xPosLerpLol[Std.int(Math.min(frameTypeBeat, xPosLerpLol.length - 1))];
+        scale.x = xFrames[frameInTypeBeat];
+        scale.y = 1 / xFrames[frameInTypeBeat];
+        x = FlxG.width * xPosLerpLol[Std.int(Math.min(frameInTypeBeat, xPosLerpLol.length - 1))];
 
-        frameTypeBeat += 1;
+        frameInTypeBeat += 1;
+      }
+    }
+
+    if (doJumpOut)
+    {
+      frameOutTicker += elapsed;
+
+      if (frameOutTicker >= 1 / 24 && frameOutTypeBeat < xFrames.length)
+      {
+        frameOutTicker = 0;
+
+        scale.x = xFrames[frameOutTypeBeat];
+        scale.y = 1 / xFrames[frameOutTypeBeat];
+        x = FlxG.width * xPosOutLerpLol[Std.int(Math.min(frameOutTypeBeat, xPosOutLerpLol.length - 1))];
+
+        frameOutTypeBeat += 1;
       }
     }
 

From 4a6b9610468991d64602e5fe5d4b71d94850fa18 Mon Sep 17 00:00:00 2001
From: Cameron Taylor <cameron.taylor.ninja@gmail.com>
Date: Thu, 16 Mar 2023 00:17:52 -0400
Subject: [PATCH 2/5] ezier transition stuff

---
 source/funkin/FreeplayState.hx | 126 +++++++++++++++++++++------------
 1 file changed, 79 insertions(+), 47 deletions(-)

diff --git a/source/funkin/FreeplayState.hx b/source/funkin/FreeplayState.hx
index 20dca2eb7..781f817b8 100644
--- a/source/funkin/FreeplayState.hx
+++ b/source/funkin/FreeplayState.hx
@@ -71,7 +71,7 @@ class FreeplayState extends MusicBeatSubstate
   var dj:DJBoyfriend;
 
   var typing:FlxInputText;
-  var exitMovers:Map<FlxSprite, MoveData> = new Map();
+  var exitMovers:Map<Array<FlxSprite>, MoveData> = new Map();
 
   override function create()
   {
@@ -132,14 +132,6 @@ class FreeplayState extends MusicBeatSubstate
     pinkBack.color = 0xFFffd4e9; // sets it to pink!
     pinkBack.x -= pinkBack.width;
 
-    exitMovers.set(pinkBack,
-      {
-        x: -pinkBack.width,
-        y: pinkBack.y,
-        speed: 0.4,
-        wait: 0
-      });
-
     FlxTween.tween(pinkBack, {x: 0}, 0.6, {ease: FlxEase.quartOut});
     add(pinkBack);
 
@@ -149,6 +141,14 @@ class FreeplayState extends MusicBeatSubstate
     var alsoOrangeLOL:FlxSprite = new FlxSprite(0, orangeBackShit.y).makeGraphic(100, Std.int(orangeBackShit.height), 0xFFffd400);
     add(alsoOrangeLOL);
 
+    exitMovers.set([pinkBack, orangeBackShit, alsoOrangeLOL],
+      {
+        x: -pinkBack.width,
+        y: pinkBack.y,
+        speed: 0.4,
+        wait: 0
+      });
+
     FlxSpriteUtil.alphaMaskFlxSprite(orangeBackShit, pinkBack, orangeBackShit);
     orangeBackShit.visible = false;
     alsoOrangeLOL.visible = false;
@@ -162,7 +162,7 @@ class FreeplayState extends MusicBeatSubstate
     moreWays.speed = 4;
     grpTxtScrolls.add(moreWays);
 
-    exitMovers.set(moreWays,
+    exitMovers.set([moreWays],
       {
         x: FlxG.width,
         y: moreWays.y,
@@ -175,7 +175,7 @@ class FreeplayState extends MusicBeatSubstate
     funnyScroll.speed = -1;
     grpTxtScrolls.add(funnyScroll);
 
-    exitMovers.set(funnyScroll,
+    exitMovers.set([funnyScroll],
       {
         x: -funnyScroll.width,
         y: funnyScroll.y,
@@ -185,7 +185,7 @@ class FreeplayState extends MusicBeatSubstate
 
     var txtNuts:BGScrollingText = new BGScrollingText(0, 300, "PROTECT YO NUTS", FlxG.width / 2);
     grpTxtScrolls.add(txtNuts);
-    exitMovers.set(txtNuts,
+    exitMovers.set([txtNuts],
       {
         x: FlxG.width,
         y: txtNuts.y,
@@ -198,7 +198,7 @@ class FreeplayState extends MusicBeatSubstate
     funnyScroll2.speed = -1.2;
     grpTxtScrolls.add(funnyScroll2);
 
-    exitMovers.set(funnyScroll2,
+    exitMovers.set([funnyScroll2],
       {
         x: -funnyScroll2.width,
         y: funnyScroll2.y,
@@ -211,7 +211,7 @@ class FreeplayState extends MusicBeatSubstate
     moreWays2.speed = 4.4;
     grpTxtScrolls.add(moreWays2);
 
-    exitMovers.set(moreWays2,
+    exitMovers.set([moreWays2],
       {
         x: FlxG.width,
         y: moreWays2.y,
@@ -224,7 +224,7 @@ class FreeplayState extends MusicBeatSubstate
     funnyScroll3.speed = -0.8;
     grpTxtScrolls.add(funnyScroll3);
 
-    exitMovers.set(funnyScroll3,
+    exitMovers.set([funnyScroll3],
       {
         x: -funnyScroll3.width,
         y: funnyScroll3.y,
@@ -233,7 +233,7 @@ class FreeplayState extends MusicBeatSubstate
       });
 
     dj = new DJBoyfriend(0, -100);
-    exitMovers.set(dj,
+    exitMovers.set([dj],
       {
         x: -dj.width * 1.6,
         y: dj.y,
@@ -248,22 +248,14 @@ class FreeplayState extends MusicBeatSubstate
     bgDad.shader = new AngleMask();
     bgDad.visible = false;
 
-    exitMovers.set(bgDad,
-      {
-        x: FlxG.width * 1.5,
-        y: 0,
-        speed: 0.5,
-        wait: 0
-      });
-
     var blackOverlayBullshitLOLXD:FlxSprite = new FlxSprite(FlxG.width).makeGraphic(Std.int(bgDad.width), Std.int(bgDad.height), FlxColor.BLACK);
     add(blackOverlayBullshitLOLXD); // used to mask the text lol!
 
-    exitMovers.set(blackOverlayBullshitLOLXD,
+    exitMovers.set([blackOverlayBullshitLOLXD, bgDad],
       {
         x: FlxG.width * 1.5,
         y: bgDad.height,
-        speed: 0.5,
+        speed: 0.4,
         wait: 0
       });
 
@@ -281,6 +273,13 @@ class FreeplayState extends MusicBeatSubstate
     grpDifficulties = new FlxSpriteGroup(-300, 80);
     add(grpDifficulties);
 
+    exitMovers.set([grpDifficulties],
+      {
+        x: -300,
+        speed: 0.25,
+        wait: 0
+      });
+
     grpDifficulties.add(new FlxSprite().loadGraphic(Paths.image('freeplay/freeplayEasy')));
     grpDifficulties.add(new FlxSprite().loadGraphic(Paths.image('freeplay/freeplayNorm')));
     grpDifficulties.add(new FlxSprite().loadGraphic(Paths.image('freeplay/freeplayHard')));
@@ -296,21 +295,13 @@ class FreeplayState extends MusicBeatSubstate
     add(overhangStuff);
     FlxTween.tween(overhangStuff, {y: 0}, 0.3, {ease: FlxEase.quartOut});
 
-    exitMovers.set(overhangStuff,
-      {
-        y: -overhangStuff.height,
-        x: 0,
-        speed: 0.2,
-        wait: 0
-      });
-
     var fnfFreeplay:FlxText = new FlxText(0, 12, 0, "FREEPLAY", 48);
     fnfFreeplay.font = "VCR OSD Mono";
     fnfFreeplay.visible = false;
 
-    exitMovers.set(fnfFreeplay,
+    exitMovers.set([overhangStuff, fnfFreeplay],
       {
-        y: fnfFreeplay.y - 64,
+        y: -overhangStuff.height,
         x: 0,
         speed: 0.2,
         wait: 0
@@ -343,15 +334,36 @@ class FreeplayState extends MusicBeatSubstate
     txtCompletion.visible = false;
     add(txtCompletion);
 
+    exitMovers.set([fp, txtCompletion, fnfHighscoreSpr],
+      {
+        x: FlxG.width,
+        speed: 0.3
+      });
+
     dj.onIntroDone.add(function() {
       FlxTween.tween(grpDifficulties, {x: 90}, 0.6, {ease: FlxEase.quartOut});
 
-      add(new DifficultySelector(20, grpDifficulties.y - 10, false, controls));
-      add(new DifficultySelector(325, grpDifficulties.y - 10, true, controls));
+      var diffSelLeft = new DifficultySelector(20, grpDifficulties.y - 10, false, controls);
+      var diffSelRight = new DifficultySelector(325, grpDifficulties.y - 10, true, controls);
+
+      add(diffSelLeft);
+      add(diffSelRight);
+
+      exitMovers.set([diffSelLeft, diffSelRight],
+        {
+          x: -diffSelLeft.width * 2,
+          speed: 0.26
+        });
 
       var letterSort:LetterSort = new LetterSort(300, 100);
       add(letterSort);
 
+      exitMovers.set([letterSort],
+        {
+          y: -100,
+          speed: 0.3
+        });
+
       letterSort.changeSelectionCallback = (str) -> {
         switch (str)
         {
@@ -759,12 +771,23 @@ class FreeplayState extends MusicBeatSubstate
 
       var longestTimer:Float = 0;
 
-      for (spr in exitMovers.keys())
+      for (grpSpr in exitMovers.keys())
       {
-        var moveData:MoveData = exitMovers.get(spr);
-        FlxTween.tween(spr, {x: moveData.x, y: moveData.y}, moveData.speed, {ease: FlxEase.expoIn});
+        var moveData:MoveData = exitMovers.get(grpSpr);
 
-        longestTimer = Math.max(longestTimer, moveData.speed + moveData.wait);
+        for (spr in grpSpr)
+        {
+          var funnyMoveShit:MoveData = moveData;
+
+          if (moveData.x == null) funnyMoveShit.x = spr.x;
+          if (moveData.y == null) funnyMoveShit.y = spr.y;
+          if (moveData.speed == null) funnyMoveShit.speed = 0.2;
+          if (moveData.wait == null) funnyMoveShit.wait = 0;
+
+          FlxTween.tween(spr, {x: funnyMoveShit.x, y: funnyMoveShit.y}, funnyMoveShit.speed, {ease: FlxEase.expoIn});
+
+          longestTimer = Math.max(longestTimer, funnyMoveShit.speed + funnyMoveShit.wait);
+        }
       }
 
       for (caps in grpCapsules.members)
@@ -777,7 +800,16 @@ class FreeplayState extends MusicBeatSubstate
       new FlxTimer().start(longestTimer, (_) -> {
         FlxTransitionableState.skipNextTransIn = true;
         FlxTransitionableState.skipNextTransOut = true;
-        FlxG.switchState(new MainMenuState());
+        if (Type.getClass(FlxG.state) == MainMenuState)
+        {
+          close();
+        }
+        else
+        {
+          FlxG.switchState(new MainMenuState());
+        }
+        //
+        // close();
       });
     }
 
@@ -1004,8 +1036,8 @@ class SongMetadata
 
 typedef MoveData =
 {
-  var x:Float;
-  var y:Float;
-  var speed:Float;
-  var wait:Float;
+  var ?x:Float;
+  var ?y:Float;
+  var ?speed:Float;
+  var ?wait:Float;
 }

From 91724133d792ed36470e90493e2084c2f175e5b2 Mon Sep 17 00:00:00 2001
From: Cameron Taylor <cameron.taylor.ninja@gmail.com>
Date: Thu, 16 Mar 2023 00:55:25 -0400
Subject: [PATCH 3/5] lil polish stuff!

---
 source/funkin/FreeplayState.hx |  6 +++++
 source/funkin/MainMenuState.hx | 41 ++++++++++++++++++++++------------
 2 files changed, 33 insertions(+), 14 deletions(-)

diff --git a/source/funkin/FreeplayState.hx b/source/funkin/FreeplayState.hx
index 781f817b8..6bf80fa58 100644
--- a/source/funkin/FreeplayState.hx
+++ b/source/funkin/FreeplayState.hx
@@ -797,6 +797,12 @@ class FreeplayState extends MusicBeatSubstate
         caps.doJumpOut = true;
       }
 
+      if (Type.getClass(FlxG.state) == MainMenuState)
+      {
+        FlxG.state.persistentUpdate = true;
+        FlxG.state.persistentDraw = true;
+      }
+
       new FlxTimer().start(longestTimer, (_) -> {
         FlxTransitionableState.skipNextTransIn = true;
         FlxTransitionableState.skipNextTransOut = true;
diff --git a/source/funkin/MainMenuState.hx b/source/funkin/MainMenuState.hx
index a99eac686..85c91db93 100644
--- a/source/funkin/MainMenuState.hx
+++ b/source/funkin/MainMenuState.hx
@@ -88,8 +88,7 @@ class MainMenuState extends MusicBeatState
     menuItems = new MenuTypedList<AtlasMenuItem>();
     add(menuItems);
     menuItems.onChange.add(onMenuItemChange);
-    menuItems.onAcceptPress.add(function(_)
-    {
+    menuItems.onAcceptPress.add(function(_) {
       if (_.name == 'freeplay')
       {
         magenta.visible = true;
@@ -102,8 +101,7 @@ class MainMenuState extends MusicBeatState
 
     menuItems.enabled = true; // can move on intro
     createMenuItem('storymode', 'mainmenu/storymode', function() startExitState(new StoryMenuState()));
-    createMenuItem('freeplay', 'mainmenu/freeplay', function()
-    {
+    createMenuItem('freeplay', 'mainmenu/freeplay', function() {
       persistentDraw = true;
       persistentUpdate = false;
       openSubState(new FreeplayState());
@@ -114,8 +112,7 @@ class MainMenuState extends MusicBeatState
     createMenuItem('donate', 'mainmenu/donate', selectDonate, hasPopupBlocker);
     #end
 
-    createMenuItem('options', 'mainmenu/options', function()
-    {
+    createMenuItem('options', 'mainmenu/options', function() {
       startExitState(new OptionsState());
     });
 
@@ -129,8 +126,21 @@ class MainMenuState extends MusicBeatState
       menuItem.y = top + spacing * i;
     }
 
-    FlxG.cameras.reset(new SwagCamera());
-    FlxG.camera.follow(camFollow, null, 0.06);
+    resetCamStuff();
+
+    subStateClosed.add(_ -> {
+      resetCamStuff();
+    });
+
+    subStateOpened.add(sub -> {
+      if (Type.getClass(sub) == FreeplayState)
+      {
+        new FlxTimer().start(0.5, _ -> {
+          magenta.visible = false;
+        });
+      }
+    });
+
     // FlxG.camera.setScrollBounds(bg.x, bg.x + bg.width, bg.y, bg.y + bg.height * 1.2);
 
     super.create();
@@ -142,6 +152,12 @@ class MainMenuState extends MusicBeatState
     // NG.core.calls.event.logEvent('swag').send();
   }
 
+  function resetCamStuff()
+  {
+    FlxG.cameras.reset(new SwagCamera());
+    FlxG.camera.follow(camFollow, null, 0.06);
+  }
+
   function createMenuItem(name:String, atlas:String, callback:Void->Void, fireInstantly:Bool = false):Void
   {
     var item = new AtlasMenuItem(name, Paths.getSparrowAtlas(atlas), callback);
@@ -214,8 +230,7 @@ class MainMenuState extends MusicBeatState
     var onPromptClose = checkLoginStatus;
     if (onClose != null)
     {
-      onPromptClose = function()
-      {
+      onPromptClose = function() {
         checkLoginStatus();
         onClose();
       }
@@ -235,8 +250,7 @@ class MainMenuState extends MusicBeatState
   public function openPrompt(prompt:Prompt, onClose:Void->Void)
   {
     menuItems.enabled = false;
-    prompt.closeCallback = function()
-    {
+    prompt.closeCallback = function() {
       menuItems.enabled = true;
       if (onClose != null) onClose();
     }
@@ -248,8 +262,7 @@ class MainMenuState extends MusicBeatState
   {
     menuItems.enabled = false; // disable for exit
     var duration = 0.4;
-    menuItems.forEach(function(item)
-    {
+    menuItems.forEach(function(item) {
       if (menuItems.selectedIndex != item.ID)
       {
         FlxTween.tween(item, {alpha: 0}, duration, {ease: FlxEase.quadOut});

From 0d9c19d630ade6a2dacf1d1e304260dd13568053 Mon Sep 17 00:00:00 2001
From: Cameron Taylor <cameron.taylor.ninja@gmail.com>
Date: Thu, 16 Mar 2023 01:03:45 -0400
Subject: [PATCH 4/5] freeplay exit generally workin

---
 source/funkin/FreeplayState.hx | 40 ++++++++++++----------------------
 1 file changed, 14 insertions(+), 26 deletions(-)

diff --git a/source/funkin/FreeplayState.hx b/source/funkin/FreeplayState.hx
index 6bf80fa58..c43a9142c 100644
--- a/source/funkin/FreeplayState.hx
+++ b/source/funkin/FreeplayState.hx
@@ -164,10 +164,8 @@ class FreeplayState extends MusicBeatSubstate
 
     exitMovers.set([moreWays],
       {
-        x: FlxG.width,
-        y: moreWays.y,
-        speed: 0.3,
-        wait: 0
+        x: FlxG.width * 2,
+        speed: 0.4,
       });
 
     var funnyScroll:BGScrollingText = new BGScrollingText(0, 250, "BOYFRIEND", FlxG.width / 2);
@@ -177,9 +175,9 @@ class FreeplayState extends MusicBeatSubstate
 
     exitMovers.set([funnyScroll],
       {
-        x: -funnyScroll.width,
+        x: -funnyScroll.width * 2,
         y: funnyScroll.y,
-        speed: 0.3,
+        speed: 0.4,
         wait: 0
       });
 
@@ -187,10 +185,8 @@ class FreeplayState extends MusicBeatSubstate
     grpTxtScrolls.add(txtNuts);
     exitMovers.set([txtNuts],
       {
-        x: FlxG.width,
-        y: txtNuts.y,
-        speed: 0.3,
-        wait: 0
+        x: FlxG.width * 2,
+        speed: 0.4,
       });
 
     var funnyScroll2:BGScrollingText = new BGScrollingText(0, 340, "BOYFRIEND", FlxG.width / 2);
@@ -200,10 +196,8 @@ class FreeplayState extends MusicBeatSubstate
 
     exitMovers.set([funnyScroll2],
       {
-        x: -funnyScroll2.width,
-        y: funnyScroll2.y,
-        speed: 0.3,
-        wait: 0
+        x: -funnyScroll2.width * 2,
+        speed: 0.5,
       });
 
     var moreWays2:BGScrollingText = new BGScrollingText(0, 400, "HOT BLOODED IN MORE WAYS THAN ONE", FlxG.width);
@@ -213,10 +207,8 @@ class FreeplayState extends MusicBeatSubstate
 
     exitMovers.set([moreWays2],
       {
-        x: FlxG.width,
-        y: moreWays2.y,
-        speed: 0.3,
-        wait: 0
+        x: FlxG.width * 2,
+        speed: 0.4
       });
 
     var funnyScroll3:BGScrollingText = new BGScrollingText(0, orangeBackShit.y, "BOYFRIEND", FlxG.width / 2);
@@ -226,19 +218,15 @@ class FreeplayState extends MusicBeatSubstate
 
     exitMovers.set([funnyScroll3],
       {
-        x: -funnyScroll3.width,
-        y: funnyScroll3.y,
-        speed: 0.3,
-        wait: 0
+        x: -funnyScroll3.width * 2,
+        speed: 0.3
       });
 
     dj = new DJBoyfriend(0, -100);
     exitMovers.set([dj],
       {
         x: -dj.width * 1.6,
-        y: dj.y,
-        speed: 0.5,
-        wait: 0
+        speed: 0.5
       });
     add(dj);
 
@@ -431,7 +419,7 @@ class FreeplayState extends MusicBeatSubstate
     FlxG.cameras.add(funnyCam);
 
     typing = new FlxInputText(100, 100);
-    add(typing);
+    // add(typing);
 
     typing.callback = function(txt, action) {
       // generateSongList(new EReg(txt.trim(), "ig"));

From 0905b2c4b1de26c2ae979c1517c6a4dec885dfbb Mon Sep 17 00:00:00 2001
From: Cameron Taylor <cameron.taylor.ninja@gmail.com>
Date: Thu, 16 Mar 2023 22:02:56 -0400
Subject: [PATCH 5/5] fixed exit bugs

---
 source/funkin/FreeplayState.hx | 4 ++++
 1 file changed, 4 insertions(+)

diff --git a/source/funkin/FreeplayState.hx b/source/funkin/FreeplayState.hx
index c43a9142c..622625f16 100644
--- a/source/funkin/FreeplayState.hx
+++ b/source/funkin/FreeplayState.hx
@@ -753,6 +753,10 @@ class FreeplayState extends MusicBeatSubstate
 
     if (controls.BACK && !typing.hasFocus)
     {
+      FlxTween.globalManager.clear();
+      FlxTimer.globalManager.clear();
+      dj.onIntroDone.removeAll();
+
       FlxG.sound.play(Paths.sound('cancelMenu'));
 
       // FlxTween.tween(dj, {x: -dj.width}, 0.2, {ease: FlxEase.quartOut});