|
|
- using Mono.Cecil;
- using Newtonsoft.Json.Linq;
- using System;
- using System.Collections.Generic;
- using System.Reflection;
- using System.Text;
- #if NET3
- using Net3_Proxy;
- #endif
-
- namespace IPA.Loader.Features
- {
- /// <summary>
- /// The root interface for a mod Feature.
- /// </summary>
- /// <remarks>
- /// Avoid storing any data in any subclasses. If you do, it may result in a failure to load the feature.
- /// </remarks>
- public abstract class Feature
- {
- /// <summary>
- /// Initializes the feature with the data provided in the definition.
- /// </summary>
- /// <remarks>
- /// <para>This gets called AFTER your <c>Init</c> method, but BEFORE the target's <c>Init</c> method. If it is applied to the defining plugin, <c>BeforeInit</c> is not called.</para>
- /// <para>Returning <see langword="false" /> does <i>not</i> prevent the plugin from being loaded. It simply prevents the feature from being used.</para>
- /// </remarks>
- /// <param name="meta">the metadata of the plugin that is being prepared</param>
- /// <param name="featureData">the data provided with the feature</param>
- /// <returns><see langword="true"/> if the feature is valid for the plugin, <see langword="false"/> otherwise</returns>
- protected abstract bool Initialize(PluginMetadata meta, JObject featureData);
-
- /// <summary>
- /// The message to be logged when the feature is not valid for a plugin.
- /// This should also be set whenever either <see cref="BeforeInit"/> returns false.
- /// </summary>
- /// <value>the message to show when the feature is marked invalid</value>
- public virtual string InvalidMessage { get; protected set; }
-
- /// <summary>
- /// Called before a plugin's `Init` method is called. This will not be called if there is no `Init` method. This should never throw an exception. An exception will abort the loading of the plugin with an error.
- /// </summary>
- /// <param name="plugin">the plugin to be initialized</param>
- /// <returns>whether or not to call the Init method</returns>
- public virtual bool BeforeInit(PluginMetadata plugin) => true;
-
- /// <summary>
- /// Called after a plugin has been fully initialized, whether or not there is an `Init` method. This should never throw an exception.
- /// </summary>
- /// <param name="plugin">the plugin that was just initialized</param>
- /// <param name="pluginInstance">the instance of the plugin being initialized</param>
- public virtual void AfterInit(PluginMetadata plugin, object pluginInstance) => AfterInit(plugin);
-
- /// <summary>
- /// Called after a plugin has been fully initialized, whether or not there is an `Init` method. This should never throw an exception.
- /// </summary>
- /// <param name="plugin">the plugin that was just initialized</param>
- public virtual void AfterInit(PluginMetadata plugin) { }
-
- /// <summary>
- /// Called after a plugin with this feature appplied is disabled.
- /// </summary>
- /// <param name="plugin">the plugin that was disabled</param>
- public virtual void AfterDisable(PluginMetadata plugin) { }
-
- // TODO: rework features to take arguments as JSON objects
-
- static Feature()
- {
- Reset();
- }
-
- internal static void Reset()
- {
- featureTypes = new Dictionary<string, Type>
- {
- { "IPA.DefineFeature", typeof(DefineFeature) }
- };
- featureDelcarers = new Dictionary<string, PluginMetadata>
- {
- { "IPA.DefineFeature", null }
- };
- }
-
- private static Dictionary<string, Type> featureTypes;
- private static Dictionary<string, PluginMetadata> featureDelcarers;
-
- internal static bool HasFeature(string name) => featureTypes.ContainsKey(name);
-
- internal static bool PreregisterFeature(PluginMetadata defining, string name)
- {
- if (featureDelcarers.ContainsKey(name)) return false;
- featureDelcarers.Add(name, defining);
- return true;
- }
-
- internal static bool RegisterFeature(PluginMetadata definingPlugin, string name, Type type)
- {
- if (!typeof(Feature).IsAssignableFrom(type))
- throw new ArgumentException($"Feature type not subclass of {nameof(Feature)}", nameof(type));
-
- if (featureTypes.ContainsKey(name)) return false;
-
- if (featureDelcarers.TryGetValue(name, out var declarer))
- {
- if (definingPlugin != declarer)
- return false;
- }
-
- featureTypes.Add(name, type);
- return true;
- }
-
- private class EmptyFeature : Feature
- {
- protected override bool Initialize(PluginMetadata meta, JObject featureData)
- {
- throw new NotImplementedException();
- }
- }
-
- internal string FeatureName;
-
- internal class Instance
- {
- public readonly PluginMetadata AppliedTo;
- public readonly string Name;
- public readonly JObject Data;
-
- public Instance(PluginMetadata appliedTo, string name, JObject data)
- {
- AppliedTo = appliedTo;
- Name = name;
- Data = data;
- type = null;
- }
-
- private Type type;
- public bool TryGetDefiningPlugin(out PluginMetadata plugin)
- {
- return featureDelcarers.TryGetValue(Name, out plugin);
- }
-
- // returns whether or not Initialize returned true, feature is always set when the thing exists
- public bool TryCreate(out Feature feature)
- {
- feature = null;
- if (type == null)
- {
- if (!featureTypes.TryGetValue(Name, out type))
- return false;
- }
-
- bool result;
- try
- {
- feature = (Feature)Activator.CreateInstance(type);
- feature.FeatureName = Name;
-
- result = feature.Initialize(AppliedTo, Data);
- }
- catch (Exception e)
- {
- result = false;
- feature = new EmptyFeature() { InvalidMessage = e.ToString() };
- }
- return result;
- }
- }
-
- /*
- // returns false with both outs null for no such feature
- internal static bool TryParseFeature(string featureString, PluginMetadata plugin,
- out Feature feature, out Exception failException, out bool featureValid, out Instance parsed,
- Instance? preParsed = null)
- {
- failException = null;
- feature = null;
- featureValid = false;
-
- if (preParsed == null)
- {
- var builder = new StringBuilder();
- string name = null;
- var parameters = new List<string>();
-
- bool escape = false;
- int parens = 0;
- bool removeWhitespace = true;
- foreach (var chr in featureString)
- {
- if (escape)
- {
- builder.Append(chr);
- escape = false;
- }
- else
- {
- switch (chr)
- {
- case '\\':
- escape = true;
- break;
- case '(':
- parens++;
- if (parens != 1) goto default;
- removeWhitespace = true;
- name = builder.ToString();
- builder.Clear();
- break;
- case ')':
- parens--;
- if (parens != 0) goto default;
- goto case ',';
- case ',':
- if (parens > 1) goto default;
- parameters.Add(builder.ToString());
- builder.Clear();
- removeWhitespace = true;
- break;
- default:
- if (removeWhitespace && !char.IsWhiteSpace(chr))
- removeWhitespace = false;
- if (!removeWhitespace)
- builder.Append(chr);
- break;
- }
- }
- }
-
- if (name == null)
- name = builder.ToString();
-
- parsed = new Instance(name, parameters.ToArray());
-
- if (parens != 0)
- {
- failException = new Exception("Malformed feature definition");
- return false;
- }
- }
- else
- parsed = preParsed.Value;
-
- if (!featureTypes.TryGetValue(parsed.Name, out var featureType))
- return false;
-
- try
- {
- if (!(Activator.CreateInstance(featureType) is Feature aFeature))
- {
- failException = new InvalidCastException("Feature type not a subtype of Feature");
- return false;
- }
-
- featureValid = aFeature.Initialize(plugin, TODO);
- feature = aFeature;
- return true;
- }
- catch (Exception e)
- {
- failException = e;
- return false;
- }
- }*/
- }
- }
|