From 5ef398c309b710236fea2ee9e6d8b168da159159 Mon Sep 17 00:00:00 2001 From: James Groom Date: Thu, 23 Nov 2023 06:47:36 +0000 Subject: [PATCH] Lazily init `BasicApiProvider`/`ApiContainer` --- src/BizHawk.Client.Common/Api/ApiInjector.cs | 15 +++---- .../tools/ToolManager.cs | 39 +++++++++++++------ 2 files changed, 35 insertions(+), 19 deletions(-) diff --git a/src/BizHawk.Client.Common/Api/ApiInjector.cs b/src/BizHawk.Client.Common/Api/ApiInjector.cs index 352fee3566..7fa0c877ce 100644 --- a/src/BizHawk.Client.Common/Api/ApiInjector.cs +++ b/src/BizHawk.Client.Common/Api/ApiInjector.cs @@ -35,7 +35,7 @@ namespace BizHawk.Client.Common /// Feeds the target its required Apis. /// /// false if update failed - public static bool UpdateApis(IExternalApiProvider source, object target) + public static bool UpdateApis(Func getProvider, object target) { Type targetType = target.GetType(); object[] tmp = new object[1]; @@ -44,12 +44,14 @@ namespace BizHawk.Client.Common .Where(static pi => pi.PropertyType == typeof(ApiContainer)) .Select(static pi => pi.SetMethod)) { - mi?.Invoke(target, new object[] { source.Container }); + if (mi is null) continue; + tmp[0] ??= getProvider().Container; + mi.Invoke(target, tmp); } foreach (var propinfo in targetType.GetPropertiesWithAttrib(typeof(RequiredApiAttribute))) { - tmp[0] = source.GetApi(propinfo.PropertyType); + tmp[0] = getProvider().GetApi(propinfo.PropertyType); if (tmp[0] == null) { return false; @@ -60,7 +62,7 @@ namespace BizHawk.Client.Common foreach (var propinfo in targetType.GetPropertiesWithAttrib(typeof(OptionalApiAttribute))) { - tmp[0] = source.GetApi(propinfo.PropertyType); + tmp[0] = getProvider().GetApi(propinfo.PropertyType); propinfo.GetSetMethod(true).Invoke(target, tmp); } @@ -71,11 +73,10 @@ namespace BizHawk.Client.Common /// Determines whether a target is available, considering its dependencies /// and the Apis provided by the emulator core. /// - public static bool IsAvailable(IExternalApiProvider source, Type targetType) + public static bool IsAvailable(Func getProvider, Type targetType) { return targetType.GetPropertiesWithAttrib(typeof(RequiredApiAttribute)) - .Select(pi => pi.PropertyType) - .All(source.HasApi); + .All(pi => getProvider().HasApi(pi.PropertyType)); } } diff --git a/src/BizHawk.Client.EmuHawk/tools/ToolManager.cs b/src/BizHawk.Client.EmuHawk/tools/ToolManager.cs index 45a4e1cac9..90257f1b3c 100644 --- a/src/BizHawk.Client.EmuHawk/tools/ToolManager.cs +++ b/src/BizHawk.Client.EmuHawk/tools/ToolManager.cs @@ -24,7 +24,9 @@ namespace BizHawk.Client.EmuHawk private readonly DisplayManager _displayManager; private readonly ExternalToolManager _extToolManager; private readonly InputManager _inputManager; - private IExternalApiProvider _apiProvider; + + private IExternalApiProvider _apiProvider = null; + private IEmulator _emulator; private readonly IMovieSession _movieSession; private IGameInfo _game; @@ -34,12 +36,6 @@ namespace BizHawk.Client.EmuHawk // Also a UsesRam, and similar method private readonly List _tools = new List(); - private IExternalApiProvider ApiProvider - { - get => _apiProvider; - set => _apiProvider = value; - } - /// /// Initializes a new instance of the class. /// @@ -61,9 +57,20 @@ namespace BizHawk.Client.EmuHawk _emulator = emulator; _movieSession = movieSession; _game = game; - ApiProvider = ApiManager.Restart(_emulator.ServiceProvider, _owner, _displayManager, _inputManager, _movieSession, this, _config, _emulator, _game); } + private IExternalApiProvider GetOrInitApiProvider() + => _apiProvider ??= ApiManager.Restart( + _emulator.ServiceProvider, + _owner, + _displayManager, + _inputManager, + _movieSession, + this, + _config, + _emulator, + _game); + /// /// Loads the tool dialog T (T must implements ) , if it does not exist it will be created, if it is already open, it will be focused /// This method should be used only if you can't use the generic one @@ -165,7 +172,11 @@ namespace BizHawk.Client.EmuHawk var newTool = (IExternalToolForm) CreateInstance(typeof(IExternalToolForm), toolPath, customFormTypeName, skipExtToolWarning: skipExtToolWarning); if (newTool == null) return null; if (newTool is Form form) form.Owner = _owner; - if (!(ServiceInjector.UpdateServices(_emulator.ServiceProvider, newTool) && ApiInjector.UpdateApis(ApiProvider, newTool))) return null; + if (!ServiceInjector.UpdateServices(_emulator.ServiceProvider, newTool) + || !ApiInjector.UpdateApis(GetOrInitApiProvider, newTool)) + { + return null; + } SetBaseProperties(newTool); // auto settings if (newTool is IToolFormAutoConfig autoConfigTool) @@ -557,7 +568,7 @@ namespace BizHawk.Client.EmuHawk _config = config; _emulator = emulator; _game = game; - ApiProvider = ApiManager.Restart(_emulator.ServiceProvider, _owner, _displayManager, _inputManager, _movieSession, this, _config, _emulator, _game); + _apiProvider = null; // If Cheat tool is loaded, restarting will restart the list too anyway if (!Has()) { @@ -570,7 +581,7 @@ namespace BizHawk.Client.EmuHawk { SetBaseProperties(tool); if (ServiceInjector.UpdateServices(_emulator.ServiceProvider, tool) - && (tool is not IExternalToolForm || ApiInjector.UpdateApis(ApiProvider, tool))) + && (tool is not IExternalToolForm || ApiInjector.UpdateApis(GetOrInitApiProvider, tool))) { if (tool.IsActive) tool.Restart(); } @@ -758,7 +769,11 @@ namespace BizHawk.Client.EmuHawk public bool IsAvailable(Type tool) { if (!ServiceInjector.IsAvailable(_emulator.ServiceProvider, tool)) return false; - if (typeof(IExternalToolForm).IsAssignableFrom(tool) && !ApiInjector.IsAvailable(ApiProvider, tool)) return false; + if (typeof(IExternalToolForm).IsAssignableFrom(tool) + && !ApiInjector.IsAvailable(GetOrInitApiProvider, tool)) + { + return false; + } if (!PossibleToolTypeNames.Contains(tool.AssemblyQualifiedName) && !_extToolManager.PossibleExtToolTypeNames.Contains(tool.AssemblyQualifiedName)) return false; // not a tool ToolAttribute attr = tool.GetCustomAttributes(false).OfType().SingleOrDefault();