ingame launcher for fnf mod states
Just drag modlauncher.zip
into your mods folder and you're good to go!
When in the main menu, press TAB
to open the launcher UI and your BACK
key to close it. While it's open, you'll see banners of any mods that have bound to the launcher. You can navigate between them with your UI_LEFT
and UI_RIGHT
keys, and press your ACCEPT
key to "launch" the selected mod. You can also cancel the selection by pressing BACK
before the state transition starts.
The following documentation is also available in the modlauncher.Registry
code, but here it is for your convenience:
To bind your mod, you must call bind
from the modlauncher.Registry
module and pass in an anonymous struct which requires the following fields of LauncherData
:
Field | Type | Description |
---|---|---|
name |
String |
The name of your mod as it will appear in the launcher. |
target |
String |
Full class name of the ScriptedMusicBeatState you want to open after your mod is selected. For example, "exampleMod.states.InitState" . |
logoPath |
String |
The path to your mod's logo (will be passed into Paths.image ).Defaults to the modlauncher icon. |
selectSoundPath |
Null<String> |
The path to your mod's select sound (will be passed into Paths.sound ).Defaults to "confirmMenu" . |
selectDuration |
Null<Float> |
The time in seconds from when the player selcts your mod to the end of the state transition, minimum 1.5 .The selection is cancellable until 0.5 seconds before this duration (when the state transition starts).Defaults to 1.5 . |
onSetup |
Null<(LauncherData)->Void> |
Callback to run when modlauncher is injected into the main menu. Useful for setting up your mod banner in the launcher. |
onUpdate |
Null<(LauncherData, Float)->Void> |
Callback to run every frame while the launcher is open. Useful for animating your banner The 2nd parameter is delta-time in seconds. |
onFocus |
Null<(LauncherData)->Void> |
Callback to run when your mod is "focused" on. |
onUnfocus |
Null<(LauncherData)->Void> |
Callback to run when another mod is focused on away from yours. |
onSelect |
Null<(LauncherData)->Void> |
Callback to run when your mod is initially selected, before the state transition starts. |
onCancel |
Null<(LauncherData)->Void> |
Callback to run when your mod selection is cancelled before the state transition starts. |
onInit |
Null<(LauncherData)->Void> |
Callback to run after your mod is selected, and right before your target state is initialized. Useful for "initializing" your mod if you're not using an initialization state (I recommend the latter, though). |
The LauncherData
parameter passed into the callbacks is the same as the struct you passed into bind
, but with a few more fields used for the banner:
Field | Type | Description |
---|---|---|
camera |
FunkinCamera |
The camera of your mod's banner in the launcher. The size you have to work with is 1280x450. |
groupBG |
FlxTypedSpriteGroup |
The background group of your mod's banner. |
groupUI |
FlxTypedSpriteGroup |
The UI group of your mod's banner. |
logo |
FunkinSprite |
The logo sprite of your mod's banner, part of groupUI .Initially auto-resized to 350px height. |
So, if you add to this struct in one of your callbacks, it will carry over into the other callbacks.
Example binding:
package exampleMod;
import flixel.tweens.FlxEase;
import flixel.tweens.FlxTween;
import funkin.graphics.FunkinSprite;
import funkin.modding.module.ModuleHandler;
import funkin.modding.module.ScriptedModule;
class LauncherBinding extends ScriptedModule {
public function new() {
super("exampleMod.LauncherBinding");
}
override public function onCreate(event:ScriptEvent):Void {
active = false;
tryBindModlauncher();
// Bind to other FNF mod launchers as well? Up to you...
hook();
}
private function hook():Void {
ModuleHandler.getModule("cynlib.reloader.Reloader").scriptGet("reloadPre").set("exampleMod.LauncherBinding", {
callback: "onReload"
});
}
public function onReload():Void {
tryBindModlauncher();
// Bind to other FNF mod launchers as well? Up to you...
hook();
}
private var bound:Bool = false;
public function tryBindModlauncher():Void {
if (bound) {
return;
}
if (ModuleHandler.getModule("modlauncher.Registry") != null) {
ModuleHandler.getModule("modlauncher.Registry").scriptCall("bind", [{
name: "Example Mod",
target: "exampleMod.states.InitState",
logoPath: "exampleMod/logo",
selectSoundPath: "exampleMod/launcherSelectSound",
selectSoundLength: 2.5,
onSetup: function(data:Dynamic):Void {
data.camera.bgColor = 0xff808080;
data.logo.scale.set(0.5, 0.5);
data.logo.updateHitbox();
var mySprite:FunkinSprite = new FunkinSprite().loadTexture("exampleMod/mySprite");
data.groupBG.add(mySprite);
data.mySprite = mySprite;
},
onSelect: function(data:Dynamic):Void {
data.camera.flash(0xffffffff, 0.5, null, true);
FlxTween.globalManager.cancelTweensOf(data.logo.scale, ["x", "y"]);
data.logo.scale.set(0.45, 0.45);
FlxTween.tween(data.logo.scale, {x: 0.75, y: 0.75}, 1.5, {ease: FlxEase.expoOut});
},
onCancel: function(data:Dynamic):Void {
FlxTween.globalManager.cancelTweensOf(data.logo.scale, ["x", "y"]);
FlxTween.tween(data.logo.scale, {x: 0.5, y: 0.5}, 0.5, {ease: FlxEase.expoOut});
},
onUpdate: function(data:Dynamic, elapsed:Float):Void {
data.mySprite.angle += 90 * elapsed;
},
onInit: function(data:Dynamic):Void {
ModuleHandler.getModule("exampleMod.Globals").scriptSet("myVariable", true);
}
}]);
bound = true;
}
}
}
Important
Bound mods do not persist through polymod reload! You must re-bind your mods using the cynlib.reloader.Reloader
module like in the example above.