2022-03-11 06:30:01 +00:00
|
|
|
package funkin.modding.module;
|
|
|
|
|
2023-07-19 05:29:43 +00:00
|
|
|
import funkin.util.SortUtil;
|
2023-10-26 09:46:22 +00:00
|
|
|
import funkin.modding.events.ScriptEvent.UpdateScriptEvent;
|
2022-07-22 20:11:18 +00:00
|
|
|
import funkin.modding.events.ScriptEvent;
|
|
|
|
import funkin.modding.events.ScriptEventDispatcher;
|
|
|
|
import funkin.modding.module.Module;
|
|
|
|
import funkin.modding.module.ScriptedModule;
|
2022-03-11 06:30:01 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Utility functions for loading and manipulating active modules.
|
|
|
|
*/
|
|
|
|
class ModuleHandler
|
|
|
|
{
|
2023-01-23 00:55:30 +00:00
|
|
|
static final moduleCache:Map<String, Module> = new Map<String, Module>();
|
|
|
|
static var modulePriorityOrder:Array<String> = [];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Parses and preloads the game's stage data and scripts when the game starts.
|
2023-06-08 20:30:45 +00:00
|
|
|
*
|
2023-01-23 00:55:30 +00:00
|
|
|
* If you want to force stages to be reloaded, you can just call this function again.
|
|
|
|
*/
|
|
|
|
public static function loadModuleCache():Void
|
|
|
|
{
|
|
|
|
// Clear any stages that are cached if there were any.
|
|
|
|
clearModuleCache();
|
|
|
|
trace("[MODULEHANDLER] Loading module cache...");
|
|
|
|
|
|
|
|
var scriptedModuleClassNames:Array<String> = ScriptedModule.listScriptClasses();
|
|
|
|
trace(' Instantiating ${scriptedModuleClassNames.length} modules...');
|
|
|
|
for (moduleCls in scriptedModuleClassNames)
|
|
|
|
{
|
|
|
|
var module:Module = ScriptedModule.init(moduleCls, moduleCls);
|
|
|
|
if (module != null)
|
|
|
|
{
|
|
|
|
trace(' Loaded module: ${moduleCls}');
|
|
|
|
|
|
|
|
// Then store it.
|
|
|
|
addToModuleCache(module);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
trace(' Failed to instantiate module: ${moduleCls}');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
reorderModuleCache();
|
|
|
|
|
|
|
|
trace("[MODULEHANDLER] Module cache loaded.");
|
|
|
|
}
|
|
|
|
|
|
|
|
public static function buildModuleCallbacks():Void
|
|
|
|
{
|
|
|
|
FlxG.signals.postStateSwitch.add(onStateSwitchComplete);
|
|
|
|
}
|
|
|
|
|
|
|
|
static function onStateSwitchComplete():Void
|
|
|
|
{
|
2023-10-26 09:46:22 +00:00
|
|
|
callEvent(new StateChangeScriptEvent(STATE_CHANGE_END, FlxG.state, true));
|
2023-01-23 00:55:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static function addToModuleCache(module:Module):Void
|
|
|
|
{
|
|
|
|
moduleCache.set(module.moduleId, module);
|
|
|
|
}
|
|
|
|
|
|
|
|
static function reorderModuleCache():Void
|
|
|
|
{
|
|
|
|
modulePriorityOrder = moduleCache.keys().array();
|
|
|
|
|
2023-08-22 08:27:30 +00:00
|
|
|
modulePriorityOrder.sort(sortByPriority);
|
|
|
|
}
|
2023-01-23 00:55:30 +00:00
|
|
|
|
2023-08-22 08:27:30 +00:00
|
|
|
/**
|
|
|
|
* Given two module IDs, sort them by priority.
|
|
|
|
* @return 1 or -1 depending on which module has a higher priority.
|
|
|
|
*/
|
|
|
|
static function sortByPriority(a:String, b:String)
|
|
|
|
{
|
|
|
|
var aModule:Module = moduleCache.get(a);
|
|
|
|
var bModule:Module = moduleCache.get(b);
|
|
|
|
|
|
|
|
if (aModule.priority != bModule.priority)
|
|
|
|
{
|
|
|
|
return aModule.priority - bModule.priority;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return SortUtil.alphabetically(a, b);
|
|
|
|
}
|
2023-01-23 00:55:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public static function getModule(moduleId:String):Module
|
|
|
|
{
|
|
|
|
return moduleCache.get(moduleId);
|
|
|
|
}
|
|
|
|
|
|
|
|
public static function activateModule(moduleId:String):Void
|
|
|
|
{
|
|
|
|
var module:Module = getModule(moduleId);
|
|
|
|
if (module != null)
|
|
|
|
{
|
|
|
|
module.active = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static function deactivateModule(moduleId:String):Void
|
|
|
|
{
|
|
|
|
var module:Module = getModule(moduleId);
|
|
|
|
if (module != null)
|
|
|
|
{
|
|
|
|
module.active = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Clear the module cache, forcing all modules to call shutdown events.
|
|
|
|
*/
|
|
|
|
public static function clearModuleCache():Void
|
|
|
|
{
|
|
|
|
if (moduleCache != null)
|
|
|
|
{
|
2023-10-26 09:46:22 +00:00
|
|
|
var event = new ScriptEvent(DESTROY, false);
|
2023-01-23 00:55:30 +00:00
|
|
|
|
|
|
|
// Note: Ignore stopPropagation()
|
|
|
|
for (key => value in moduleCache)
|
|
|
|
{
|
|
|
|
ScriptEventDispatcher.callEvent(value, event);
|
|
|
|
}
|
|
|
|
|
|
|
|
moduleCache.clear();
|
|
|
|
modulePriorityOrder = [];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static function callEvent(event:ScriptEvent):Void
|
|
|
|
{
|
|
|
|
for (moduleId in modulePriorityOrder)
|
|
|
|
{
|
|
|
|
var module:Module = moduleCache.get(moduleId);
|
|
|
|
// The module needs to be active to receive events.
|
|
|
|
if (module != null && module.active)
|
|
|
|
{
|
|
|
|
ScriptEventDispatcher.callEvent(module, event);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static inline function callOnCreate():Void
|
|
|
|
{
|
2023-10-26 09:46:22 +00:00
|
|
|
callEvent(new ScriptEvent(CREATE, false));
|
2023-01-23 00:55:30 +00:00
|
|
|
}
|
2022-03-11 06:30:01 +00:00
|
|
|
}
|