Funkin/source/funkin/ui/freeplay/FreeplayState.hx

1479 lines
40 KiB
Haxe
Raw Normal View History

package funkin.ui.freeplay;
2021-04-07 20:19:49 -04:00
import flixel.addons.transition.FlxTransitionableState;
import flixel.addons.ui.FlxInputText;
2021-08-21 20:45:03 -04:00
import flixel.FlxCamera;
2020-10-21 14:05:27 -04:00
import flixel.FlxSprite;
2021-10-21 23:08:48 -04:00
import flixel.group.FlxGroup;
import flixel.group.FlxGroup.FlxTypedGroup;
2024-03-11 23:42:32 -04:00
import flixel.group.FlxSpriteGroup.FlxTypedSpriteGroup;
2021-08-21 19:53:08 -04:00
import flixel.input.touch.FlxTouch;
2021-08-24 15:28:04 -04:00
import flixel.math.FlxAngle;
import flixel.math.FlxPoint;
import flixel.system.debug.watch.Tracker.TrackerProfile;
2020-10-21 14:05:27 -04:00
import flixel.text.FlxText;
2021-10-21 23:08:48 -04:00
import flixel.tweens.FlxEase;
2021-03-02 00:46:28 -05:00
import flixel.tweens.FlxTween;
2020-11-06 21:17:27 -05:00
import flixel.util.FlxColor;
2021-10-21 23:08:48 -04:00
import flixel.util.FlxSpriteUtil;
2021-10-21 17:40:53 -04:00
import flixel.util.FlxTimer;
2024-03-11 23:42:32 -04:00
import funkin.audio.FunkinSound;
2024-03-23 15:34:37 -04:00
import funkin.data.story.level.LevelRegistry;
import funkin.data.song.SongRegistry;
2024-03-11 23:42:32 -04:00
import funkin.graphics.FunkinCamera;
import funkin.graphics.FunkinSprite;
import funkin.graphics.shaders.AngleMask;
import funkin.graphics.shaders.HSVShader;
import funkin.graphics.shaders.PureColor;
import funkin.graphics.shaders.StrokeShader;
import funkin.input.Controls;
import funkin.play.PlayStatePlaylist;
import funkin.play.song.Song;
import funkin.save.Save;
import funkin.save.Save.SaveScoreData;
import funkin.ui.AtlasText;
import funkin.ui.mainmenu.MainMenuState;
import funkin.ui.MusicBeatSubState;
import funkin.ui.transition.LoadingState;
import funkin.ui.transition.StickerSubState;
import funkin.util.MathUtil;
import lime.utils.Assets;
2020-10-21 14:05:27 -04:00
2024-02-05 13:35:30 -05:00
/**
* Parameters used to initialize the FreeplayState.
*/
typedef FreeplayStateParams =
{
?character:String,
2024-02-05 13:35:30 -05:00
};
/**
* The state for the freeplay menu, allowing the player to select any song to play.
*/
class FreeplayState extends MusicBeatSubState
2020-10-21 14:05:27 -04:00
{
//
// Params
//
/**
* The current character for this FreeplayState.
* You can't change this without transitioning to a new FreeplayState.
*/
final currentCharacter:String;
2024-02-09 14:58:57 -05:00
/**
* For the audio preview, the duration of the fade-in effect.
*/
public static final FADE_IN_DURATION:Float = 0.5;
/**
* For the audio preview, the duration of the fade-out effect.
*/
public static final FADE_OUT_DURATION:Float = 0.25;
/**
* For the audio preview, the volume at which the fade-in starts.
*/
public static final FADE_IN_START_VOLUME:Float = 0.25;
/**
* For the audio preview, the volume at which the fade-in ends.
*/
public static final FADE_IN_END_VOLUME:Float = 1.0;
/**
* For the audio preview, the volume at which the fade-out starts.
*/
public static final FADE_OUT_END_VOLUME:Float = 0.0;
var songs:Array<Null<FreeplaySongData>> = [];
var diffIdsCurrent:Array<String> = [];
var diffIdsTotal:Array<String> = [];
2020-10-21 14:05:27 -04:00
2023-01-22 19:55:30 -05:00
var curSelected:Int = 0;
var currentDifficulty:String = Constants.DEFAULT_DIFFICULTY;
2020-11-06 21:17:27 -05:00
2023-01-22 19:55:30 -05:00
var fp:FreeplayScore;
var txtCompletion:AtlasText;
2023-01-22 19:55:30 -05:00
var lerpCompletion:Float = 0;
var intendedCompletion:Float = 0;
var lerpScore:Float = 0;
var intendedScore:Int = 0;
2020-10-21 14:05:27 -04:00
var grpDifficulties:FlxTypedSpriteGroup<DifficultySprite>;
2021-12-07 17:41:18 -05:00
2023-01-22 19:55:30 -05:00
var coolColors:Array<Int> = [
0xFF9271FD,
0xFF9271FD,
0xFF223344,
2023-01-22 19:55:30 -05:00
0xFF941653,
0xFFFC96D7,
0xFFA0D1FF,
0xFFFF78BF,
0xFFF6B604
2023-01-22 19:55:30 -05:00
];
2021-03-02 00:46:28 -05:00
var grpSongs:FlxTypedGroup<Alphabet>;
var grpCapsules:FlxTypedGroup<SongMenuItem>;
2023-09-28 20:29:19 -04:00
var curCapsule:SongMenuItem;
var curPlaying:Bool = false;
2020-11-01 14:16:22 -05:00
var displayedVariations:Array<String>;
var dj:DJBoyfriend;
var ostName:FlxText;
var albumRoll:AlbumRoll;
var letterSort:LetterSort;
var exitMovers:ExitMoverData = new Map();
2022-09-26 18:22:45 -04:00
2023-04-06 01:39:27 -04:00
var stickerSubState:StickerSubState;
2024-04-01 21:59:53 -04:00
public static var rememberedDifficulty:Null<String> = Constants.DEFAULT_DIFFICULTY;
public static var rememberedSongId:Null<String> = 'tutorial';
public function new(?params:FreeplayStateParams, ?stickers:StickerSubState)
2023-04-06 01:39:27 -04:00
{
2024-02-05 13:35:30 -05:00
currentCharacter = params?.character ?? Constants.DEFAULT_CHARACTER;
2023-04-06 01:39:27 -04:00
if (stickers != null)
{
stickerSubState = stickers;
}
super(FlxColor.TRANSPARENT);
2023-04-06 01:39:27 -04:00
}
override function create():Void
2023-01-22 19:55:30 -05:00
{
2023-04-06 01:39:27 -04:00
super.create();
2024-05-02 23:02:47 -04:00
FlxG.state.persistentUpdate = false;
2023-01-22 19:55:30 -05:00
FlxTransitionableState.skipNextTransIn = true;
2021-10-21 23:08:48 -04:00
2023-04-06 01:39:27 -04:00
if (stickerSubState != null)
{
this.persistentUpdate = true;
this.persistentDraw = true;
openSubState(stickerSubState);
stickerSubState.degenStickers();
}
2023-01-22 19:55:30 -05:00
#if discord_rpc
// Updating Discord Rich Presence
DiscordClient.changePresence('In the Menus', null);
2023-01-22 19:55:30 -05:00
#end
2021-05-06 06:54:57 -04:00
2023-01-22 19:55:30 -05:00
var isDebug:Bool = false;
2021-05-06 06:54:57 -04:00
2023-01-22 19:55:30 -05:00
#if debug
isDebug = true;
#end
2021-05-06 06:54:57 -04:00
FunkinSound.playMusic('freakyMenu',
{
overrideExisting: true,
restartTrack: false
});
2020-11-01 14:16:22 -05:00
// Add a null entry that represents the RANDOM option
songs.push(null);
2021-03-01 18:59:51 -05:00
// TODO: This makes custom variations disappear from Freeplay. Figure out a better solution later.
// Default character (BF) shows default and Erect variations. Pico shows only Pico variations.
displayedVariations = (currentCharacter == 'bf') ? [Constants.DEFAULT_VARIATION, 'erect'] : [currentCharacter];
2023-09-28 20:29:19 -04:00
// programmatically adds the songs via LevelRegistry and SongRegistry
for (levelId in LevelRegistry.instance.listSortedLevelIds())
2023-09-28 20:29:19 -04:00
{
for (songId in LevelRegistry.instance.parseEntryData(levelId).songs)
2023-09-28 20:29:19 -04:00
{
var song:Song = SongRegistry.instance.fetchEntry(songId);
// Only display songs which actually have available charts for the current character.
var availableDifficultiesForSong:Array<String> = song.listDifficulties(displayedVariations, false);
if (availableDifficultiesForSong.length == 0) continue;
songs.push(new FreeplaySongData(levelId, songId, song, displayedVariations));
for (difficulty in availableDifficultiesForSong)
{
diffIdsTotal.pushUnique(difficulty);
}
2023-09-28 20:29:19 -04:00
}
}
2021-09-06 14:50:04 -04:00
2023-01-22 19:55:30 -05:00
// LOAD MUSIC
2020-10-21 14:05:27 -04:00
2023-01-22 19:55:30 -05:00
// LOAD CHARACTERS
2020-10-21 14:05:27 -04:00
2023-01-22 19:55:30 -05:00
trace(FlxG.width);
trace(FlxG.camera.zoom);
trace(FlxG.camera.initialZoom);
trace(FlxCamera.defaultZoom);
2020-10-25 16:51:06 -04:00
var pinkBack:FunkinSprite = FunkinSprite.create('freeplay/pinkBack');
pinkBack.color = 0xFFFFD4E9; // sets it to pink!
2023-01-22 19:55:30 -05:00
pinkBack.x -= pinkBack.width;
2021-10-21 23:08:48 -04:00
2023-01-22 19:55:30 -05:00
FlxTween.tween(pinkBack, {x: 0}, 0.6, {ease: FlxEase.quartOut});
add(pinkBack);
2021-10-21 23:08:48 -04:00
var orangeBackShit:FunkinSprite = new FunkinSprite(84, 440).makeSolidColor(Std.int(pinkBack.width), 75, 0xFFFEDA00);
2023-01-22 19:55:30 -05:00
add(orangeBackShit);
2021-10-21 23:08:48 -04:00
var alsoOrangeLOL:FunkinSprite = new FunkinSprite(0, orangeBackShit.y).makeSolidColor(100, Std.int(orangeBackShit.height), 0xFFFFD400);
2023-01-22 19:55:30 -05:00
add(alsoOrangeLOL);
2021-10-21 23:08:48 -04:00
2023-03-16 00:17:52 -04:00
exitMovers.set([pinkBack, orangeBackShit, alsoOrangeLOL],
{
x: -pinkBack.width,
y: pinkBack.y,
speed: 0.4,
wait: 0
});
2023-01-22 19:55:30 -05:00
FlxSpriteUtil.alphaMaskFlxSprite(orangeBackShit, pinkBack, orangeBackShit);
orangeBackShit.visible = false;
alsoOrangeLOL.visible = false;
2021-10-21 23:08:48 -04:00
2023-01-22 19:55:30 -05:00
var grpTxtScrolls:FlxGroup = new FlxGroup();
add(grpTxtScrolls);
grpTxtScrolls.visible = false;
2021-10-21 23:08:48 -04:00
FlxG.debugger.addTrackerProfile(new TrackerProfile(BGScrollingText, ['x', 'y', 'speed', 'size']));
var moreWays:BGScrollingText = new BGScrollingText(0, 160, 'HOT BLOODED IN MORE WAYS THAN ONE', FlxG.width, true, 43);
moreWays.funnyColor = 0xFFFFF383;
moreWays.speed = 6.8;
2023-01-22 19:55:30 -05:00
grpTxtScrolls.add(moreWays);
2021-10-21 23:08:48 -04:00
2023-03-16 00:17:52 -04:00
exitMovers.set([moreWays],
2023-03-15 21:05:15 -04:00
{
2023-03-16 01:03:45 -04:00
x: FlxG.width * 2,
speed: 0.4,
2023-03-15 21:05:15 -04:00
});
var funnyScroll:BGScrollingText = new BGScrollingText(0, 220, 'BOYFRIEND', FlxG.width / 2, false, 60);
funnyScroll.funnyColor = 0xFFFF9963;
funnyScroll.speed = -3.8;
2023-01-22 19:55:30 -05:00
grpTxtScrolls.add(funnyScroll);
2021-10-21 23:08:48 -04:00
2023-03-16 00:17:52 -04:00
exitMovers.set([funnyScroll],
2023-03-15 21:05:15 -04:00
{
2023-03-16 01:03:45 -04:00
x: -funnyScroll.width * 2,
2023-03-15 21:05:15 -04:00
y: funnyScroll.y,
2023-03-16 01:03:45 -04:00
speed: 0.4,
2023-03-15 21:05:15 -04:00
wait: 0
});
var txtNuts:BGScrollingText = new BGScrollingText(0, 285, 'PROTECT YO NUTS', FlxG.width / 2, true, 43);
txtNuts.speed = 3.5;
2023-01-22 19:55:30 -05:00
grpTxtScrolls.add(txtNuts);
2023-03-16 00:17:52 -04:00
exitMovers.set([txtNuts],
2023-03-15 21:05:15 -04:00
{
2023-03-16 01:03:45 -04:00
x: FlxG.width * 2,
speed: 0.4,
2023-03-15 21:05:15 -04:00
});
2021-10-21 23:08:48 -04:00
var funnyScroll2:BGScrollingText = new BGScrollingText(0, 335, 'BOYFRIEND', FlxG.width / 2, false, 60);
funnyScroll2.funnyColor = 0xFFFF9963;
funnyScroll2.speed = -3.8;
2023-01-22 19:55:30 -05:00
grpTxtScrolls.add(funnyScroll2);
2021-10-21 23:08:48 -04:00
2023-03-16 00:17:52 -04:00
exitMovers.set([funnyScroll2],
2023-03-15 21:05:15 -04:00
{
2023-03-16 01:03:45 -04:00
x: -funnyScroll2.width * 2,
speed: 0.5,
2023-03-15 21:05:15 -04:00
});
var moreWays2:BGScrollingText = new BGScrollingText(0, 397, 'HOT BLOODED IN MORE WAYS THAN ONE', FlxG.width, true, 43);
moreWays2.funnyColor = 0xFFFFF383;
moreWays2.speed = 6.8;
2023-01-22 19:55:30 -05:00
grpTxtScrolls.add(moreWays2);
2023-03-16 00:17:52 -04:00
exitMovers.set([moreWays2],
2023-03-15 21:05:15 -04:00
{
2023-03-16 01:03:45 -04:00
x: FlxG.width * 2,
speed: 0.4
2023-03-15 21:05:15 -04:00
});
var funnyScroll3:BGScrollingText = new BGScrollingText(0, orangeBackShit.y + 10, 'BOYFRIEND', FlxG.width / 2, 60);
funnyScroll3.funnyColor = 0xFFFEA400;
funnyScroll3.speed = -3.8;
2023-01-22 19:55:30 -05:00
grpTxtScrolls.add(funnyScroll3);
2023-03-16 00:17:52 -04:00
exitMovers.set([funnyScroll3],
2023-03-15 21:05:15 -04:00
{
2023-03-16 01:03:45 -04:00
x: -funnyScroll3.width * 2,
speed: 0.3
2023-03-15 21:05:15 -04:00
});
dj = new DJBoyfriend(640, 366);
2023-03-16 00:17:52 -04:00
exitMovers.set([dj],
2023-03-15 21:05:15 -04:00
{
x: -dj.width * 1.6,
2023-03-16 01:03:45 -04:00
speed: 0.5
2023-03-15 21:05:15 -04:00
});
// TODO: Replace this.
if (currentCharacter == 'pico') dj.visible = false;
2023-01-22 19:55:30 -05:00
add(dj);
var bgDad:FlxSprite = new FlxSprite(pinkBack.width * 0.75, 0).loadGraphic(Paths.image('freeplay/freeplayBGdad'));
bgDad.setGraphicSize(0, FlxG.height);
bgDad.updateHitbox();
bgDad.shader = new AngleMask();
bgDad.visible = false;
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!
2023-03-16 00:17:52 -04:00
exitMovers.set([blackOverlayBullshitLOLXD, bgDad],
2023-03-15 21:05:15 -04:00
{
x: FlxG.width * 1.5,
2023-03-16 00:17:52 -04:00
speed: 0.4,
2023-03-15 21:05:15 -04:00
wait: 0
});
2023-01-22 19:55:30 -05:00
add(bgDad);
2024-04-30 01:38:00 -04:00
FlxTween.tween(blackOverlayBullshitLOLXD, {x: pinkBack.width * 0.75}, 0.7, {ease: FlxEase.quintOut});
2023-01-22 19:55:30 -05:00
blackOverlayBullshitLOLXD.shader = bgDad.shader;
grpSongs = new FlxTypedGroup<Alphabet>();
add(grpSongs);
grpCapsules = new FlxTypedGroup<SongMenuItem>();
add(grpCapsules);
grpDifficulties = new FlxTypedSpriteGroup<DifficultySprite>(-300, 80);
2023-01-22 19:55:30 -05:00
add(grpDifficulties);
2023-03-16 00:17:52 -04:00
exitMovers.set([grpDifficulties],
{
x: -300,
speed: 0.25,
wait: 0
});
for (diffId in diffIdsTotal)
{
var diffSprite:DifficultySprite = new DifficultySprite(diffId);
diffSprite.difficultyId = diffId;
grpDifficulties.add(diffSprite);
}
2023-01-22 19:55:30 -05:00
2023-03-15 21:05:15 -04:00
grpDifficulties.group.forEach(function(spr) {
2023-01-22 19:55:30 -05:00
spr.visible = false;
});
for (diffSprite in grpDifficulties.group.members)
{
if (diffSprite == null) continue;
if (diffSprite.difficultyId == currentDifficulty) diffSprite.visible = true;
}
2023-01-22 19:55:30 -05:00
albumRoll = new AlbumRoll();
albumRoll.albumId = null;
add(albumRoll);
2023-08-09 19:34:19 -04:00
albumRoll.applyExitMovers(exitMovers);
2023-08-09 19:34:19 -04:00
2023-01-22 19:55:30 -05:00
var overhangStuff:FlxSprite = new FlxSprite().makeGraphic(FlxG.width, 64, FlxColor.BLACK);
overhangStuff.y -= overhangStuff.height;
add(overhangStuff);
FlxTween.tween(overhangStuff, {y: 0}, 0.3, {ease: FlxEase.quartOut});
var fnfFreeplay:FlxText = new FlxText(8, 8, 0, 'FREEPLAY', 48);
fnfFreeplay.font = 'VCR OSD Mono';
2023-01-22 19:55:30 -05:00
fnfFreeplay.visible = false;
2023-03-15 21:05:15 -04:00
ostName = new FlxText(8, 8, FlxG.width - 8 - 8, 'OFFICIAL OST', 48);
ostName.font = 'VCR OSD Mono';
ostName.alignment = RIGHT;
ostName.visible = false;
2024-01-11 00:52:42 -05:00
exitMovers.set([overhangStuff, fnfFreeplay, ostName],
2023-03-15 21:05:15 -04:00
{
2023-03-16 00:17:52 -04:00
y: -overhangStuff.height,
2023-03-15 21:05:15 -04:00
x: 0,
speed: 0.2,
wait: 0
});
var sillyStroke:StrokeShader = new StrokeShader(0xFFFFFFFF, 2, 2);
2023-01-22 19:55:30 -05:00
fnfFreeplay.shader = sillyStroke;
2024-04-30 03:22:30 -04:00
ostName.shader = sillyStroke;
2023-01-22 19:55:30 -05:00
add(fnfFreeplay);
2024-01-11 00:52:42 -05:00
add(ostName);
2023-01-22 19:55:30 -05:00
var fnfHighscoreSpr:FlxSprite = new FlxSprite(860, 70);
2023-01-22 19:55:30 -05:00
fnfHighscoreSpr.frames = Paths.getSparrowAtlas('freeplay/highscore');
fnfHighscoreSpr.animation.addByPrefix('highscore', 'highscore small instance 1', 24, false);
2023-01-22 19:55:30 -05:00
fnfHighscoreSpr.visible = false;
fnfHighscoreSpr.setGraphicSize(0, Std.int(fnfHighscoreSpr.height * 1));
fnfHighscoreSpr.updateHitbox();
add(fnfHighscoreSpr);
2023-03-15 21:05:15 -04:00
new FlxTimer().start(FlxG.random.float(12, 50), function(tmr) {
fnfHighscoreSpr.animation.play('highscore');
2023-01-22 19:55:30 -05:00
tmr.time = FlxG.random.float(20, 60);
}, 0);
2024-04-03 04:52:12 -04:00
fp = new FreeplayScore(460, 60, 7, 100);
2023-01-22 19:55:30 -05:00
fp.visible = false;
add(fp);
var clearBoxSprite:FlxSprite = new FlxSprite(1165, 65).loadGraphic(Paths.image('freeplay/clearBox'));
2024-04-30 01:33:29 -04:00
clearBoxSprite.visible = false;
add(clearBoxSprite);
txtCompletion = new AtlasText(1185, 87, '69', AtlasFont.FREEPLAY_CLEAR);
2023-01-22 19:55:30 -05:00
txtCompletion.visible = false;
add(txtCompletion);
letterSort = new LetterSort(400, 75);
2023-08-09 16:15:34 -04:00
add(letterSort);
letterSort.visible = false;
exitMovers.set([letterSort],
{
y: -100,
speed: 0.3
});
letterSort.changeSelectionCallback = (str) -> {
switch (str)
{
case 'fav':
2023-08-09 16:15:34 -04:00
generateSongList({filterType: FAVORITE}, true);
case 'ALL':
2023-08-09 19:34:19 -04:00
generateSongList(null, true);
2023-08-09 16:15:34 -04:00
default:
generateSongList({filterType: REGEXP, filterData: str}, true);
}
// We want to land on the first song of the group, rather than random song when changing letter sorts
// that is, only if there's more than one song in the group!
if (grpCapsules.members.length > 0)
{
curSelected = 1;
changeSelection();
}
2023-08-09 16:15:34 -04:00
};
2024-04-30 01:33:29 -04:00
exitMovers.set([fp, txtCompletion, fnfHighscoreSpr, txtCompletion, clearBoxSprite],
2023-03-16 00:17:52 -04:00
{
x: FlxG.width,
speed: 0.3
});
var diffSelLeft:DifficultySelector = new DifficultySelector(20, grpDifficulties.y - 10, false, controls);
var diffSelRight:DifficultySelector = new DifficultySelector(325, grpDifficulties.y - 10, true, controls);
diffSelLeft.visible = false;
diffSelRight.visible = false;
add(diffSelLeft);
add(diffSelRight);
// be careful not to "add()" things in here unless it's to a group that's already added to the state
// otherwise it won't be properly attatched to funnyCamera (relavent code should be at the bottom of create())
2023-03-15 21:05:15 -04:00
dj.onIntroDone.add(function() {
2023-08-09 19:34:19 -04:00
// when boyfriend hits dat shiii
2023-09-19 23:27:07 -04:00
albumRoll.playIntro();
2023-08-09 19:34:19 -04:00
new FlxTimer().start(0.75, function(_) {
2024-04-05 19:47:33 -04:00
// albumRoll.showTitle();
});
2023-01-22 19:55:30 -05:00
FlxTween.tween(grpDifficulties, {x: 90}, 0.6, {ease: FlxEase.quartOut});
diffSelLeft.visible = true;
diffSelRight.visible = true;
2023-08-09 16:15:34 -04:00
letterSort.visible = true;
2023-03-16 00:17:52 -04:00
exitMovers.set([diffSelLeft, diffSelRight],
{
x: -diffSelLeft.width * 2,
speed: 0.26
});
2023-01-22 19:55:30 -05:00
2023-03-15 21:05:15 -04:00
new FlxTimer().start(1 / 24, function(handShit) {
2023-01-22 19:55:30 -05:00
fnfHighscoreSpr.visible = true;
fnfFreeplay.visible = true;
2024-01-11 00:52:42 -05:00
ostName.visible = true;
2023-01-22 19:55:30 -05:00
fp.visible = true;
fp.updateScore(0);
2024-04-30 01:33:29 -04:00
clearBoxSprite.visible = true;
2023-01-22 19:55:30 -05:00
txtCompletion.visible = true;
intendedCompletion = 0;
2023-03-15 21:05:15 -04:00
new FlxTimer().start(1.5 / 24, function(bold) {
2023-01-22 19:55:30 -05:00
sillyStroke.width = 0;
sillyStroke.height = 0;
2023-09-19 23:27:07 -04:00
changeSelection();
2023-01-22 19:55:30 -05:00
});
});
pinkBack.color = 0xFFFFD863;
2023-01-22 19:55:30 -05:00
bgDad.visible = true;
orangeBackShit.visible = true;
alsoOrangeLOL.visible = true;
grpTxtScrolls.visible = true;
});
2023-08-09 02:47:22 -04:00
generateSongList(null, false);
2023-01-22 19:55:30 -05:00
// dedicated camera for the state so we don't need to fuk around with camera scrolls from the mainmenu / elsewhere
var funnyCam:FunkinCamera = new FunkinCamera('freeplayFunny', 0, 0, FlxG.width, FlxG.height);
2023-01-22 19:55:30 -05:00
funnyCam.bgColor = FlxColor.TRANSPARENT;
FlxG.cameras.add(funnyCam, false);
2023-01-22 19:55:30 -05:00
2023-03-15 21:05:15 -04:00
forEach(function(bs) {
2023-01-22 19:55:30 -05:00
bs.cameras = [funnyCam];
});
}
var currentFilter:SongFilter = null;
var currentFilteredSongs:Array<FreeplaySongData> = [];
/**
* Given the current filter, rebuild the current song list.
*
* @param filterStuff A filter to apply to the song list (regex, startswith, all, favorite)
2024-04-16 22:12:07 -04:00
* @param force Whether the capsules should "jump" back in or not using their animation
* @param onlyIfChanged Only apply the filter if the song list has changed
*/
public function generateSongList(filterStuff:Null<SongFilter>, force:Bool = false, onlyIfChanged:Bool = true):Void
2023-01-22 19:55:30 -05:00
{
var tempSongs:Array<FreeplaySongData> = songs;
2023-01-22 19:55:30 -05:00
// Remember just the difficulty because it's important for song sorting.
if (rememberedDifficulty != null)
{
currentDifficulty = rememberedDifficulty;
}
2024-04-16 22:12:07 -04:00
if (filterStuff != null) tempSongs = sortSongs(tempSongs, filterStuff);
2023-01-22 19:55:30 -05:00
// Filter further by current selected difficulty.
if (currentDifficulty != null)
{
tempSongs = tempSongs.filter(song -> {
if (song == null) return true; // Random
return song.songDifficulties.contains(currentDifficulty);
});
}
if (onlyIfChanged)
{
// == performs equality by reference
if (tempSongs.isEqualUnordered(currentFilteredSongs)) return;
}
// Only now do we know that the filter is actually changing.
// If curSelected is 0, the result will be null and fall back to the rememberedSongId.
rememberedSongId = grpCapsules.members[curSelected]?.songData?.songId ?? rememberedSongId;
for (cap in grpCapsules.members)
{
cap.kill();
}
currentFilter = filterStuff;
currentFilteredSongs = tempSongs;
curSelected = 0;
var hsvShader:HSVShader = new HSVShader();
2023-08-09 02:47:22 -04:00
var randomCapsule:SongMenuItem = grpCapsules.recycle(SongMenuItem);
randomCapsule.init(FlxG.width, 0, null);
2023-08-06 16:24:34 -04:00
randomCapsule.onConfirm = function() {
capsuleOnConfirmRandom(randomCapsule);
2023-08-06 16:24:34 -04:00
};
randomCapsule.y = randomCapsule.intendedY(0) + 10;
randomCapsule.targetPos.x = randomCapsule.x;
randomCapsule.alpha = 0.5;
randomCapsule.songText.visible = false;
randomCapsule.favIcon.visible = false;
randomCapsule.initJumpIn(0, force);
randomCapsule.hsvShader = hsvShader;
2023-08-06 16:24:34 -04:00
grpCapsules.add(randomCapsule);
2023-01-22 19:55:30 -05:00
for (i in 0...tempSongs.length)
{
if (tempSongs[i] == null) continue;
2023-08-09 02:47:22 -04:00
var funnyMenu:SongMenuItem = grpCapsules.recycle(SongMenuItem);
funnyMenu.init(FlxG.width, 0, tempSongs[i]);
2023-09-28 20:29:19 -04:00
funnyMenu.onConfirm = function() {
capsuleOnConfirmDefault(funnyMenu);
};
2023-08-04 17:10:27 -04:00
funnyMenu.y = funnyMenu.intendedY(i + 1) + 10;
2023-01-22 19:55:30 -05:00
funnyMenu.targetPos.x = funnyMenu.x;
funnyMenu.ID = i;
2023-09-19 19:10:30 -04:00
funnyMenu.capsule.alpha = 0.5;
2023-01-22 19:55:30 -05:00
funnyMenu.songText.visible = false;
funnyMenu.favIcon.visible = tempSongs[i].isFav;
funnyMenu.hsvShader = hsvShader;
2023-01-22 19:55:30 -05:00
funnyMenu.forcePosition();
2023-01-22 19:55:30 -05:00
grpCapsules.add(funnyMenu);
}
FlxG.console.registerFunction('changeSelection', changeSelection);
2023-09-19 23:27:07 -04:00
rememberSelection();
2023-01-22 19:55:30 -05:00
changeSelection();
changeDiff(0, true);
2023-01-22 19:55:30 -05:00
}
2024-04-16 22:12:07 -04:00
/**
* Filters an array of songs based on a filter
* @param songsToFilter What data to use when filtering
* @param songFilter The filter to apply
* @return Array<FreeplaySongData>
*/
public function sortSongs(songsToFilter:Array<FreeplaySongData>, songFilter:SongFilter):Array<FreeplaySongData>
{
switch (songFilter.filterType)
{
case REGEXP:
// filterStuff.filterData has a string with the first letter of the sorting range, and the second one
// this creates a filter to return all the songs that start with a letter between those two
// if filterData looks like "A-C", the regex should look something like this: ^[A-C].*
// to get every song that starts between A and C
var filterRegexp:EReg = new EReg('^[' + songFilter.filterData + '].*', 'i');
songsToFilter = songsToFilter.filter(str -> {
if (str == null) return true; // Random
return filterRegexp.match(str.songName);
});
case STARTSWITH:
// extra note: this is essentially a "search"
songsToFilter = songsToFilter.filter(str -> {
if (str == null) return true; // Random
return str.songName.toLowerCase().startsWith(songFilter.filterData);
});
case ALL:
// no filter!
case FAVORITE:
songsToFilter = songsToFilter.filter(str -> {
if (str == null) return true; // Random
return str.isFav;
});
default:
// return all on default
}
return songsToFilter;
}
2023-01-22 19:55:30 -05:00
var touchY:Float = 0;
var touchX:Float = 0;
var dxTouch:Float = 0;
var dyTouch:Float = 0;
var velTouch:Float = 0;
var veloctiyLoopShit:Float = 0;
var touchTimer:Float = 0;
var initTouchPos:FlxPoint = new FlxPoint();
var spamTimer:Float = 0;
var spamming:Bool = false;
var busy:Bool = false; // Set to true once the user has pressed enter to select a song.
2024-03-11 23:42:32 -04:00
override function update(elapsed:Float):Void
2023-01-22 19:55:30 -05:00
{
super.update(elapsed);
if (FlxG.keys.justPressed.F)
{
var targetSong = grpCapsules.members[curSelected]?.songData;
if (targetSong != null)
2023-01-22 19:55:30 -05:00
{
var realShit:Int = curSelected;
targetSong.isFav = !targetSong.isFav;
if (targetSong.isFav)
{
FlxTween.tween(grpCapsules.members[realShit], {angle: 360}, 0.4,
{
ease: FlxEase.elasticOut,
onComplete: _ -> {
grpCapsules.members[realShit].favIcon.visible = true;
grpCapsules.members[realShit].favIcon.animation.play('fav');
}
});
}
else
{
grpCapsules.members[realShit].favIcon.animation.play('fav', false, true);
new FlxTimer().start((1 / 24) * 14, _ -> {
grpCapsules.members[realShit].favIcon.visible = false;
});
new FlxTimer().start((1 / 24) * 24, _ -> {
FlxTween.tween(grpCapsules.members[realShit], {angle: 0}, 0.4, {ease: FlxEase.elasticOut});
});
}
2023-01-22 19:55:30 -05:00
}
}
lerpScore = MathUtil.coolLerp(lerpScore, intendedScore, 0.2);
lerpCompletion = MathUtil.coolLerp(lerpCompletion, intendedCompletion, 0.9);
2023-01-22 19:55:30 -05:00
if (Math.isNaN(lerpScore))
{
lerpScore = intendedScore;
}
if (Math.isNaN(lerpCompletion))
{
lerpCompletion = intendedCompletion;
}
2023-01-22 19:55:30 -05:00
fp.updateScore(Std.int(lerpScore));
txtCompletion.text = '${Math.floor(lerpCompletion * 100)}';
// Right align the completion percentage
switch (txtCompletion.text.length)
{
case 3:
2024-04-30 01:33:29 -04:00
txtCompletion.offset.x = 10;
case 2:
2024-04-30 01:33:29 -04:00
txtCompletion.offset.x = 0;
case 1:
2024-04-30 01:33:29 -04:00
txtCompletion.offset.x = -24;
default:
2024-04-30 01:33:29 -04:00
txtCompletion.offset.x = 0;
}
2023-01-22 19:55:30 -05:00
handleInputs(elapsed);
}
function handleInputs(elapsed:Float):Void
{
if (busy) return;
2023-01-22 19:55:30 -05:00
var upP:Bool = controls.UI_UP_P && !FlxG.keys.pressed.CONTROL;
var downP:Bool = controls.UI_DOWN_P && !FlxG.keys.pressed.CONTROL;
var accepted:Bool = controls.ACCEPT && !FlxG.keys.pressed.CONTROL;
2023-01-22 19:55:30 -05:00
if (FlxG.onMobile)
{
for (touch in FlxG.touches.list)
{
if (touch.justPressed)
{
initTouchPos.set(touch.screenX, touch.screenY);
}
if (touch.pressed)
{
var dx:Float = initTouchPos.x - touch.screenX;
var dy:Float = initTouchPos.y - touch.screenY;
2023-01-22 19:55:30 -05:00
var angle:Float = Math.atan2(dy, dx);
var length:Float = Math.sqrt(dx * dx + dy * dy);
2023-01-22 19:55:30 -05:00
FlxG.watch.addQuick('LENGTH', length);
FlxG.watch.addQuick('ANGLE', Math.round(FlxAngle.asDegrees(angle)));
2023-01-22 19:55:30 -05:00
}
}
if (FlxG.touches.getFirst() != null)
{
if (touchTimer >= 1.5) accepted = true;
2023-01-22 19:55:30 -05:00
touchTimer += elapsed;
var touch:FlxTouch = FlxG.touches.getFirst();
velTouch = Math.abs((touch.screenY - dyTouch)) / 50;
dyTouch = touch.screenY - touchY;
dxTouch = touch.screenX - touchX;
if (touch.justPressed)
{
touchY = touch.screenY;
dyTouch = 0;
velTouch = 0;
touchX = touch.screenX;
dxTouch = 0;
}
if (Math.abs(dxTouch) >= 100)
{
touchX = touch.screenX;
if (dxTouch != 0) dxTouch < 0 ? changeDiff(1) : changeDiff(-1);
2023-01-22 19:55:30 -05:00
}
if (Math.abs(dyTouch) >= 100)
{
touchY = touch.screenY;
if (dyTouch != 0) dyTouch < 0 ? changeSelection(1) : changeSelection(-1);
2023-01-22 19:55:30 -05:00
}
}
else
{
touchTimer = 0;
}
}
#if mobile
for (touch in FlxG.touches.list)
{
if (touch.justPressed)
{
// accepted = true;
}
}
#end
if (!FlxG.keys.pressed.CONTROL && (controls.UI_UP || controls.UI_DOWN))
2023-01-22 19:55:30 -05:00
{
if (spamming)
{
if (spamTimer >= 0.07)
{
spamTimer = 0;
if (controls.UI_UP)
{
changeSelection(-1);
}
2023-01-22 19:55:30 -05:00
else
{
2023-01-22 19:55:30 -05:00
changeSelection(1);
}
2023-01-22 19:55:30 -05:00
}
}
else if (spamTimer >= 0.9)
{
spamming = true;
}
else if (spamTimer <= 0)
{
if (controls.UI_UP)
{
changeSelection(-1);
}
else
{
changeSelection(1);
}
}
spamTimer += elapsed;
dj.resetAFKTimer();
2023-01-22 19:55:30 -05:00
}
else
{
spamming = false;
spamTimer = 0;
}
if (FlxG.mouse.wheel != 0)
{
dj.resetAFKTimer();
changeSelection(-Math.round(FlxG.mouse.wheel / 4));
}
if (controls.UI_LEFT_P && !FlxG.keys.pressed.CONTROL)
2023-01-22 19:55:30 -05:00
{
dj.resetAFKTimer();
changeDiff(-1);
generateSongList(currentFilter, true);
2023-01-22 19:55:30 -05:00
}
if (controls.UI_RIGHT_P && !FlxG.keys.pressed.CONTROL)
2023-01-22 19:55:30 -05:00
{
dj.resetAFKTimer();
changeDiff(1);
generateSongList(currentFilter, true);
2023-01-22 19:55:30 -05:00
}
2024-05-02 21:23:55 -04:00
if (controls.BACK)
2023-01-22 19:55:30 -05:00
{
2023-03-16 22:02:56 -04:00
FlxTween.globalManager.clear();
FlxTimer.globalManager.clear();
dj.onIntroDone.removeAll();
2024-03-23 17:50:48 -04:00
FunkinSound.playOnce(Paths.sound('cancelMenu'));
2023-01-22 19:55:30 -05:00
2023-03-15 21:05:15 -04:00
var longestTimer:Float = 0;
2023-03-16 00:17:52 -04:00
for (grpSpr in exitMovers.keys())
2023-03-15 21:05:15 -04:00
{
2023-03-16 00:17:52 -04:00
var moveData:MoveData = exitMovers.get(grpSpr);
for (spr in grpSpr)
{
if (spr == null) continue;
2023-03-16 00:17:52 -04:00
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;
2023-03-15 21:05:15 -04:00
2023-03-16 00:17:52 -04:00
FlxTween.tween(spr, {x: funnyMoveShit.x, y: funnyMoveShit.y}, funnyMoveShit.speed, {ease: FlxEase.expoIn});
longestTimer = Math.max(longestTimer, funnyMoveShit.speed + funnyMoveShit.wait);
}
2023-03-15 21:05:15 -04:00
}
for (caps in grpCapsules.members)
{
caps.doJumpIn = false;
caps.doLerp = false;
caps.doJumpOut = true;
}
if (Type.getClass(_parentState) == MainMenuState)
2023-03-16 00:55:25 -04:00
{
_parentState.persistentUpdate = false;
_parentState.persistentDraw = true;
2023-03-16 00:55:25 -04:00
}
2023-03-15 21:05:15 -04:00
new FlxTimer().start(longestTimer, (_) -> {
FlxTransitionableState.skipNextTransIn = true;
FlxTransitionableState.skipNextTransOut = true;
if (Type.getClass(_parentState) == MainMenuState)
2023-03-16 00:17:52 -04:00
{
FunkinSound.playMusic('freakyMenu',
{
overrideExisting: true,
restartTrack: false
});
2023-03-16 00:17:52 -04:00
close();
}
else
{
FlxG.switchState(() -> new MainMenuState());
2023-03-16 00:17:52 -04:00
}
2023-03-15 21:05:15 -04:00
});
2023-01-22 19:55:30 -05:00
}
if (accepted)
{
2023-08-06 16:24:34 -04:00
grpCapsules.members[curSelected].onConfirm();
2023-01-22 19:55:30 -05:00
}
}
2023-12-15 21:09:01 -05:00
public override function destroy():Void
2023-01-22 19:55:30 -05:00
{
2023-12-15 21:09:01 -05:00
super.destroy();
var daSong:Null<FreeplaySongData> = currentFilteredSongs[curSelected];
if (daSong != null)
{
clearDaCache(daSong.songName);
}
2023-01-22 19:55:30 -05:00
}
function changeDiff(change:Int = 0, force:Bool = false):Void
2023-01-22 19:55:30 -05:00
{
touchTimer = 0;
var currentDifficultyIndex:Int = diffIdsCurrent.indexOf(currentDifficulty);
2023-01-22 19:55:30 -05:00
if (currentDifficultyIndex == -1) currentDifficultyIndex = diffIdsCurrent.indexOf(Constants.DEFAULT_DIFFICULTY);
2023-01-22 19:55:30 -05:00
currentDifficultyIndex += change;
if (currentDifficultyIndex < 0) currentDifficultyIndex = diffIdsCurrent.length - 1;
if (currentDifficultyIndex >= diffIdsCurrent.length) currentDifficultyIndex = 0;
currentDifficulty = diffIdsCurrent[currentDifficultyIndex];
var daSong:Null<FreeplaySongData> = grpCapsules.members[curSelected].songData;
if (daSong != null)
{
var songScore:SaveScoreData = Save.instance.getSongScore(grpCapsules.members[curSelected].songData.songId, currentDifficulty);
intendedScore = songScore?.score ?? 0;
intendedCompletion = songScore?.accuracy ?? 0.0;
rememberedDifficulty = currentDifficulty;
}
else
{
intendedScore = 0;
intendedCompletion = 0.0;
}
2023-01-22 19:55:30 -05:00
if (intendedCompletion == Math.POSITIVE_INFINITY || intendedCompletion == Math.NEGATIVE_INFINITY || Math.isNaN(intendedCompletion))
{
intendedCompletion = 0;
}
grpDifficulties.group.forEach(function(diffSprite) {
diffSprite.visible = false;
2023-01-22 19:55:30 -05:00
});
for (diffSprite in grpDifficulties.group.members)
{
if (diffSprite == null) continue;
if (diffSprite.difficultyId == currentDifficulty)
{
if (change != 0)
{
diffSprite.visible = true;
diffSprite.offset.y += 5;
diffSprite.alpha = 0.5;
new FlxTimer().start(1 / 24, function(swag) {
diffSprite.alpha = 1;
diffSprite.updateHitbox();
});
}
else
{
diffSprite.visible = true;
}
}
}
if (change != 0 || force)
{
// Update the song capsules to reflect the new difficulty info.
for (songCapsule in grpCapsules.members)
{
if (songCapsule == null) continue;
if (songCapsule.songData != null)
{
songCapsule.songData.currentDifficulty = currentDifficulty;
songCapsule.init(null, null, songCapsule.songData);
}
else
{
songCapsule.init(null, null, null);
}
}
}
// Set the album graphic and play the animation if relevant.
var newAlbumId:String = daSong?.albumId;
if (albumRoll.albumId != newAlbumId)
{
albumRoll.albumId = newAlbumId;
2024-03-30 00:54:07 -04:00
albumRoll.skipIntro();
}
2023-01-22 19:55:30 -05:00
}
// Clears the cache of songs, frees up memory, they' ll have to be loaded in later tho function clearDaCache(actualSongTho:String)
function clearDaCache(actualSongTho:String):Void
2023-01-22 19:55:30 -05:00
{
for (song in songs)
{
if (song == null) continue;
2023-01-22 19:55:30 -05:00
if (song.songName != actualSongTho)
{
trace('trying to remove: ' + song.songName);
// openfl.Assets.cache.clear(Paths.inst(song.songName));
}
}
}
function capsuleOnConfirmRandom(randomCapsule:SongMenuItem):Void
2023-08-06 16:24:34 -04:00
{
trace('RANDOM SELECTED');
2023-08-06 16:24:34 -04:00
busy = true;
letterSort.inputEnabled = false;
var availableSongCapsules:Array<SongMenuItem> = grpCapsules.members.filter(function(cap:SongMenuItem) {
// Dead capsules are ones which were removed from the list when changing filters.
return cap.alive && cap.songData != null;
});
trace('Available songs: ${availableSongCapsules.map(function(cap) {
return cap.songData.songName;
})}');
if (availableSongCapsules.length == 0)
{
trace('No songs available!');
busy = false;
letterSort.inputEnabled = true;
2024-03-23 17:50:48 -04:00
FunkinSound.playOnce(Paths.sound('cancelMenu'));
return;
}
var targetSong:SongMenuItem = FlxG.random.getObject(availableSongCapsules);
// Seeing if I can do an animation...
curSelected = grpCapsules.members.indexOf(targetSong);
changeSelection(0); // Trigger an update.
// Act like we hit Confirm on that song.
capsuleOnConfirmDefault(targetSong);
}
2023-09-28 20:29:19 -04:00
function capsuleOnConfirmDefault(cap:SongMenuItem):Void
2023-01-22 19:55:30 -05:00
{
busy = true;
letterSort.inputEnabled = false;
2023-08-06 16:24:34 -04:00
PlayStatePlaylist.isStoryMode = false;
2023-01-22 19:55:30 -05:00
2023-11-01 23:20:47 -04:00
var targetSong:Song = SongRegistry.instance.fetchEntry(cap.songData.songId);
if (targetSong == null)
{
FlxG.log.warn('WARN: could not find song with id (${cap.songData.songId})');
return;
}
var targetDifficulty:String = currentDifficulty;
var targetVariation:String = targetSong.getFirstValidVariation(targetDifficulty);
2023-08-06 16:24:34 -04:00
PlayStatePlaylist.campaignId = cap.songData.levelId;
2023-08-06 16:24:34 -04:00
// Visual and audio effects.
2024-03-23 17:50:48 -04:00
FunkinSound.playOnce(Paths.sound('confirmMenu'));
2023-08-06 16:24:34 -04:00
dj.confirm();
new FlxTimer().start(1, function(tmr:FlxTimer) {
Paths.setCurrentLevel(cap.songData.levelId);
LoadingState.loadPlayState(
2023-08-06 16:24:34 -04:00
{
targetSong: targetSong,
targetDifficulty: targetDifficulty,
targetVariation: targetVariation,
practiceMode: false,
minimalMode: false,
#if (debug || FORCE_DEBUG_VERSION)
botPlayMode: FlxG.keys.pressed.SHIFT,
#else
botPlayMode: false,
#end
// TODO: Make these an option! It's currently only accessible via chart editor.
// startTimestamp: 0.0,
// playbackRate: 0.5,
// botPlayMode: true,
}, true);
2023-08-06 16:24:34 -04:00
});
}
function rememberSelection():Void
{
if (rememberedSongId != null)
{
curSelected = currentFilteredSongs.findIndex(function(song) {
if (song == null) return false;
return song.songId == rememberedSongId;
});
if (curSelected == -1) curSelected = 0;
}
if (rememberedDifficulty != null)
{
currentDifficulty = rememberedDifficulty;
}
}
2024-03-11 23:42:32 -04:00
function changeSelection(change:Int = 0):Void
2023-01-22 19:55:30 -05:00
{
2024-03-23 17:50:48 -04:00
FunkinSound.playOnce(Paths.sound('scrollMenu'), 0.4);
2023-01-22 19:55:30 -05:00
var prevSelected:Int = curSelected;
2023-01-22 19:55:30 -05:00
curSelected += change;
2023-08-09 03:03:58 -04:00
if (curSelected < 0) curSelected = grpCapsules.countLiving() - 1;
if (curSelected >= grpCapsules.countLiving()) curSelected = 0;
2023-01-22 19:55:30 -05:00
var daSongCapsule:SongMenuItem = grpCapsules.members[curSelected];
if (daSongCapsule.songData != null)
2023-08-09 03:03:58 -04:00
{
var songScore:SaveScoreData = Save.instance.getSongScore(daSongCapsule.songData.songId, currentDifficulty);
intendedScore = songScore?.score ?? 0;
intendedCompletion = songScore?.accuracy ?? 0.0;
2023-11-01 23:20:47 -04:00
diffIdsCurrent = daSongCapsule.songData.songDifficulties;
rememberedSongId = daSongCapsule.songData.songId;
changeDiff();
2023-08-09 03:03:58 -04:00
}
else
{
intendedScore = 0;
intendedCompletion = 0.0;
diffIdsCurrent = diffIdsTotal;
rememberedSongId = null;
rememberedDifficulty = null;
albumRoll.albumId = null;
2023-08-09 03:03:58 -04:00
}
2023-01-22 19:55:30 -05:00
for (index => capsule in grpCapsules.members)
{
2023-08-04 15:59:17 -04:00
index += 1;
2023-01-22 19:55:30 -05:00
2023-09-19 23:27:07 -04:00
capsule.selected = index == curSelected + 1;
2023-01-22 19:55:30 -05:00
2023-08-04 17:10:27 -04:00
capsule.targetPos.y = capsule.intendedY(index - curSelected);
2023-01-22 19:55:30 -05:00
capsule.targetPos.x = 270 + (60 * (Math.sin(index - curSelected)));
if (index < curSelected) capsule.targetPos.y -= 100; // another 100 for good measure
2023-01-22 19:55:30 -05:00
}
2023-08-13 22:12:08 -04:00
if (grpCapsules.countLiving() > 0)
{
if (curSelected == 0)
{
FunkinSound.playMusic('freeplayRandom',
{
startingVolume: 0.0,
overrideExisting: true,
restartTrack: true
});
2023-08-13 22:12:08 -04:00
FlxG.sound.music.fadeIn(2, 0, 0.8);
}
else
{
2024-05-09 01:56:52 -04:00
var potentiallyErect:String = (currentDifficulty == "erect") || (currentDifficulty == "nightmare") ? "-erect" : "";
// TODO: Stream the instrumental of the selected song?
2024-04-18 20:23:03 -04:00
FunkinSound.playMusic(daSongCapsule.songData.songId,
{
2024-05-09 01:56:52 -04:00
startingVolume: 0.0,
overrideExisting: true,
2024-04-18 20:23:03 -04:00
restartTrack: false,
pathsFunction: INST,
2024-05-09 01:56:52 -04:00
suffix: potentiallyErect,
2024-04-18 20:23:03 -04:00
partialParams:
{
loadPartial: true,
start: 0,
end: 0.1
2024-05-09 01:56:52 -04:00
},
onLoad: function() {
FlxG.sound.music.fadeIn(2, 0, 0.4);
}
});
}
2023-08-13 22:12:08 -04:00
grpCapsules.members[curSelected].selected = true;
}
2023-01-22 19:55:30 -05:00
}
/**
* Build an instance of `FreeplayState` that is above the `MainMenuState`.
* @return The MainMenuState with the FreeplayState as a substate.
*/
public static function build(?params:FreeplayStateParams, ?stickers:StickerSubState):MusicBeatState
{
var result = new MainMenuState();
result.openSubState(new FreeplayState(params, stickers));
2024-05-01 15:22:20 -04:00
result.persistentUpdate = false;
result.persistentDraw = true;
return result;
}
2021-12-07 19:29:26 -05:00
}
/**
* The difficulty selector arrows to the left and right of the difficulty.
*/
2021-12-07 19:29:26 -05:00
class DifficultySelector extends FlxSprite
{
2023-01-22 19:55:30 -05:00
var controls:Controls;
var whiteShader:PureColor;
2021-12-07 19:29:26 -05:00
2023-01-22 19:55:30 -05:00
public function new(x:Float, y:Float, flipped:Bool, controls:Controls)
{
super(x, y);
2021-12-07 19:29:26 -05:00
2023-01-22 19:55:30 -05:00
this.controls = controls;
2021-12-07 19:29:26 -05:00
2023-01-22 19:55:30 -05:00
frames = Paths.getSparrowAtlas('freeplay/freeplaySelector');
animation.addByPrefix('shine', 'arrow pointer loop', 24);
2023-01-22 19:55:30 -05:00
animation.play('shine');
2021-12-07 19:29:26 -05:00
2023-01-22 19:55:30 -05:00
whiteShader = new PureColor(FlxColor.WHITE);
2021-12-07 22:34:01 -05:00
2023-01-22 19:55:30 -05:00
shader = whiteShader;
2021-12-07 22:34:01 -05:00
2023-01-22 19:55:30 -05:00
flipX = flipped;
}
2021-04-08 17:29:31 -04:00
2024-03-11 23:42:32 -04:00
override function update(elapsed:Float):Void
2023-01-22 19:55:30 -05:00
{
if (flipX && controls.UI_RIGHT_P && !FlxG.keys.pressed.CONTROL) moveShitDown();
if (!flipX && controls.UI_LEFT_P && !FlxG.keys.pressed.CONTROL) moveShitDown();
2021-12-07 19:29:26 -05:00
2023-01-22 19:55:30 -05:00
super.update(elapsed);
}
2021-12-07 19:29:26 -05:00
2024-03-11 23:42:32 -04:00
function moveShitDown():Void
2023-01-22 19:55:30 -05:00
{
offset.y -= 5;
2021-04-08 17:29:31 -04:00
2023-01-22 19:55:30 -05:00
whiteShader.colorSet = true;
2021-12-07 22:34:01 -05:00
2023-08-06 22:20:18 -04:00
scale.x = scale.y = 0.5;
2023-03-15 21:05:15 -04:00
new FlxTimer().start(2 / 24, function(tmr) {
2023-08-06 22:20:18 -04:00
scale.x = scale.y = 1;
2023-01-22 19:55:30 -05:00
whiteShader.colorSet = false;
updateHitbox();
});
}
2020-10-21 14:05:27 -04:00
}
2021-02-24 20:52:59 -05:00
/**
* Structure for the current song filter.
*/
2022-09-28 02:50:53 -04:00
typedef SongFilter =
{
2023-01-22 19:55:30 -05:00
var filterType:FilterType;
var ?filterData:Dynamic;
2022-09-28 02:50:53 -04:00
}
/**
* Possible types to use for the song filter.
*/
2022-09-28 02:50:53 -04:00
enum abstract FilterType(String)
{
/**
* Filter to songs which start with a string
*/
public var STARTSWITH;
/**
* Filter to songs which match a regular expression
*/
public var REGEXP;
/**
* Filter to songs which are favorited
*/
public var FAVORITE;
/**
* Filter to all songs
*/
public var ALL;
2022-09-28 02:50:53 -04:00
}
/**
* Data about a specific song in the freeplay menu.
*/
class FreeplaySongData
2021-02-24 20:52:59 -05:00
{
/**
* Whether or not the song has been favorited.
*/
public var isFav:Bool = false;
var song:Song;
public var levelId(default, null):String = '';
public var songId(default, null):String = '';
public var songDifficulties(default, null):Array<String> = [];
2023-01-22 19:55:30 -05:00
public var songName(default, null):String = '';
public var songCharacter(default, null):String = '';
public var songRating(default, null):Int = 0;
public var albumId(default, null):Null<String> = null;
public var currentDifficulty(default, set):String = Constants.DEFAULT_DIFFICULTY;
public var displayedVariations(default, null):Array<String> = [Constants.DEFAULT_VARIATION];
function set_currentDifficulty(value:String):String
{
if (currentDifficulty == value) return value;
currentDifficulty = value;
updateValues(displayedVariations);
return value;
}
public function new(levelId:String, songId:String, song:Song, ?displayedVariations:Array<String>)
2023-01-22 19:55:30 -05:00
{
this.levelId = levelId;
this.songId = songId;
this.song = song;
if (displayedVariations != null) this.displayedVariations = displayedVariations;
updateValues(displayedVariations);
}
function updateValues(variations:Array<String>):Void
{
this.songDifficulties = song.listDifficulties(variations, false, false);
if (!this.songDifficulties.contains(currentDifficulty)) currentDifficulty = Constants.DEFAULT_DIFFICULTY;
var songDifficulty:SongDifficulty = song.getDifficulty(currentDifficulty, variations);
if (songDifficulty == null) return;
this.songName = songDifficulty.songName;
this.songCharacter = songDifficulty.characters.opponent;
this.songRating = songDifficulty.difficultyRating;
if (songDifficulty.album == null)
{
FlxG.log.warn('No album for: ${songDifficulty.songName}');
this.albumId = Constants.DEFAULT_ALBUM_ID;
}
else
{
this.albumId = songDifficulty.album;
}
2023-01-22 19:55:30 -05:00
}
2021-02-24 20:52:59 -05:00
}
2023-03-15 21:05:15 -04:00
/**
* The map storing information about the exit movers.
*/
typedef ExitMoverData = Map<Array<FlxSprite>, MoveData>;
/**
* The data for an exit mover.
*/
2023-03-15 21:05:15 -04:00
typedef MoveData =
{
2023-03-16 00:17:52 -04:00
var ?x:Float;
var ?y:Float;
var ?speed:Float;
var ?wait:Float;
2023-03-15 21:05:15 -04:00
}
/**
* The sprite for the difficulty
*/
class DifficultySprite extends FlxSprite
{
/**
* The difficulty id which this sprite represents.
*/
public var difficultyId:String;
public function new(diffId:String)
{
super();
difficultyId = diffId;
if (Assets.exists(Paths.file('images/freeplay/freeplay${diffId}.xml')))
{
this.frames = Paths.getSparrowAtlas('freeplay/freeplay${diffId}');
this.animation.addByPrefix('idle', 'idle0', 24, true);
if (Preferences.flashingLights) this.animation.play('idle');
}
else
{
this.loadGraphic(Paths.image('freeplay/freeplay' + diffId));
}
}
}