/notepad-x/Plugin/PluginServices.cs

# · C# · 295 lines · 196 code · 29 blank · 70 comment · 17 complexity · b94179099ff813dcc68788f48e83e7c4 MD5 · raw file

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Text;
  4. using System.IO;
  5. using System.Reflection;
  6. namespace Notepad_X.Plugin
  7. {
  8. public class PluginServices : IPluginHost
  9. {
  10. Main main = Notepad_X.Static.MainInstance;
  11. public PluginServices()
  12. {
  13. }
  14. /// <summary>
  15. /// Available plugins collection
  16. /// </summary>
  17. public AvailablePlugins colAvailablePlugins = new AvailablePlugins();
  18. /// <summary>
  19. /// A Collection of all Plugins Found and Loaded by the FindPlugins() Method
  20. /// </summary>
  21. public AvailablePlugins AvailablePlugins
  22. {
  23. get { return colAvailablePlugins; }
  24. set { colAvailablePlugins = value; }
  25. }
  26. /// <summary>
  27. /// Searches the Application's Startup Directory for Plugins
  28. /// </summary>
  29. public void FindPlugins()
  30. {
  31. FindPlugins(AppDomain.CurrentDomain.BaseDirectory);
  32. }
  33. /// <summary>
  34. /// Searches the passed Path for Plugins
  35. /// </summary>
  36. /// <param name="Path">Directory to search for Plugins in</param>
  37. public void FindPlugins(string Path)
  38. {
  39. //First empty the collection, we're reloading them all
  40. colAvailablePlugins.Clear();
  41. //Go through all the files in the plugin directory
  42. foreach (string fileOn in Directory.GetFiles(Path))
  43. {
  44. FileInfo file = new FileInfo(fileOn);
  45. //Preliminary check, must be .dll
  46. if (file.Extension.Equals(".dll"))
  47. {
  48. //Add the 'plugin'
  49. this.AddPlugin(fileOn);
  50. }
  51. }
  52. }
  53. /// <summary>
  54. /// Unloads and Closes all AvailablePlugins
  55. /// </summary>
  56. ///
  57. public void ClosePlugin(String pluginNameOrPath)
  58. {
  59. AvailablePlugin tmp = null;
  60. foreach (AvailablePlugin pluginOn in colAvailablePlugins)
  61. {
  62. if((pluginOn.Instance.Name.Equals(pluginNameOrPath)) || pluginOn.AssemblyPath.Equals(pluginNameOrPath))
  63. {
  64. pluginOn.Instance.Dispose();
  65. pluginOn.Instance = null;
  66. tmp = pluginOn;
  67. break;
  68. }
  69. }
  70. if (tmp != null)
  71. {
  72. remove(tmp);
  73. }
  74. }
  75. private void remove(AvailablePlugin pl)
  76. {
  77. colAvailablePlugins.Remove(pl);
  78. }
  79. /// <summary>
  80. /// Close all the plugins
  81. /// </summary>
  82. public void ClosePlugins()
  83. {
  84. foreach (AvailablePlugin pluginOn in colAvailablePlugins)
  85. {
  86. try
  87. {
  88. //Close all plugin instances
  89. //We call the plugins Dispose sub first incase it has to do
  90. //Its own cleanup stuff
  91. pluginOn.Instance.Dispose();
  92. //After we give the plugin a chance to tidy up, get rid of it
  93. pluginOn.Instance = null;
  94. }catch(Exception){
  95. }
  96. }
  97. //Finally, clear our collection of available plugins
  98. colAvailablePlugins.Clear();
  99. }
  100. /// <summary>
  101. /// Reads an toolbar file and returns the properties as array (Name,Author,Version,Description)
  102. /// </summary>
  103. /// <returns>String array (Name,Author,Version,Description)</returns>
  104. /// <param name="FileName">Filename of the plugin</param>
  105. public string[] hash(string FileName)
  106. {
  107. string[] ret = null;
  108. try
  109. {
  110. Assembly pluginAssembly = Assembly.LoadFrom(FileName);
  111. foreach (Type pluginType in pluginAssembly.GetTypes())
  112. {
  113. if (pluginType.IsPublic)
  114. {
  115. if (!pluginType.IsAbstract)
  116. {
  117. Type typeInterface = pluginType.GetInterface("Notepad_X.Plugin.IPlugin", true);
  118. if (typeInterface != null)
  119. {
  120. AvailablePlugin newPlugin = new AvailablePlugin();
  121. newPlugin.AssemblyPath = FileName;
  122. newPlugin.Instance = (IPlugin)Activator.CreateInstance(pluginAssembly.GetType(pluginType.ToString()));
  123. ret = new string[4];
  124. ret[0] = newPlugin.Instance.Name;
  125. ret[1] = newPlugin.Instance.Author;
  126. ret[2] = newPlugin.Instance.Version;
  127. ret[3] = newPlugin.Instance.Description;
  128. newPlugin.Instance = null;
  129. newPlugin = null;
  130. }
  131. typeInterface = null;
  132. }
  133. }
  134. }
  135. pluginAssembly = null;
  136. }
  137. catch
  138. {
  139. ret = null;
  140. }
  141. return ret;
  142. }
  143. public void AddPlugin(string FileName)
  144. {
  145. //Create a new assembly from the plugin file we're adding..
  146. Assembly pluginAssembly = Assembly.LoadFrom(FileName);
  147. //Next we'll loop through all the Types found in the assembly
  148. foreach (Type pluginType in pluginAssembly.GetTypes())
  149. {
  150. if (pluginType.IsPublic) //Only look at public types
  151. {
  152. if (!pluginType.IsAbstract) //Only look at non-abstract types
  153. {
  154. //Gets a type object of the interface we need the plugins to match
  155. Type typeInterface = pluginType.GetInterface("Notepad_X.Plugin.IPlugin", true);
  156. //Make sure the interface we want to use actually exists
  157. if (typeInterface != null)
  158. {
  159. //Create a new available plugin since the type implements the IPlugin interface
  160. AvailablePlugin newPlugin = new AvailablePlugin();
  161. //Set the filename where we found it
  162. newPlugin.AssemblyPath = FileName;
  163. //Create a new instance and store the instance in the collection for later use
  164. //We could change this later on to not load an instance.. we have 2 options
  165. //1- Make one instance, and use it whenever we need it.. it's always there
  166. //2- Don't make an instance, and instead make an instance whenever we use it, then close it
  167. //For now we'll just make an instance of all the plugins
  168. newPlugin.Instance = (IPlugin)Activator.CreateInstance(pluginAssembly.GetType(pluginType.ToString()));
  169. //Add the new plugin to our collection here
  170. this.colAvailablePlugins.Add(newPlugin);
  171. //Call the initialization sub of the plugin
  172. newPlugin.Instance.Initialize();
  173. //cleanup
  174. newPlugin = null;
  175. }
  176. typeInterface = null; // Clean
  177. }
  178. }
  179. }
  180. pluginAssembly = null; //more cleanup
  181. }
  182. }
  183. public class AvailablePlugins : System.Collections.CollectionBase
  184. {
  185. //A Simple class to hold some info about our Available Plugins
  186. /// <summary>
  187. /// Add a Plugin to the collection of Available plugins
  188. /// </summary>
  189. /// <param name="pluginToAdd">The Plugin to Add</param>
  190. public void Add(AvailablePlugin pluginToAdd)
  191. {
  192. this.List.Add(pluginToAdd);
  193. }
  194. public string get(int id)
  195. {
  196. return ((AvailablePlugin)this.List[id]).AssemblyPath;
  197. }
  198. /// <summary>
  199. /// Remove a Plugin to the collection of Available plugins
  200. /// </summary>
  201. /// <param name="pluginToRemove">The Plugin to Remove</param>
  202. public void Remove(AvailablePlugin pluginToRemove)
  203. {
  204. this.List.Remove(pluginToRemove);
  205. pluginToRemove = null;
  206. }
  207. public bool Exist(string pluginNameOrPath)
  208. {
  209. foreach (AvailablePlugin pluginOn in this.List)
  210. {
  211. try
  212. {
  213. if ((pluginOn.Instance.Name.Equals(pluginNameOrPath)) || pluginOn.AssemblyPath.Equals(pluginNameOrPath))
  214. {
  215. return true;
  216. }
  217. }
  218. catch { }
  219. }
  220. return false;
  221. }
  222. /// <summary>
  223. /// Finds a plugin in the available Plugins
  224. /// </summary>
  225. /// <param name="pluginNameOrPath">The name or File path of the plugin to find</param>
  226. /// <returns>Available Plugin, or null if the plugin is not found</returns>
  227. public AvailablePlugin Find(string pluginNameOrPath)
  228. {
  229. //Loop through all the plugins
  230. foreach (AvailablePlugin pluginOn in this.List)
  231. {
  232. try
  233. {
  234. //Find the one with the matching name or filename
  235. if ((pluginOn.Instance.Name.Equals(pluginNameOrPath)) || pluginOn.AssemblyPath.Equals(pluginNameOrPath))
  236. {
  237. return pluginOn;
  238. }
  239. }
  240. catch { }
  241. }
  242. return null;
  243. }
  244. }
  245. /// <summary>
  246. /// Data Class for Available Plugin. Holds and instance of the loaded Plugin, as well as the Plugin's Assembly Path
  247. /// </summary>
  248. public class AvailablePlugin
  249. {
  250. //This is the actual AvailablePlugin object..
  251. //Holds an instance of the plugin to access
  252. //Also holds assembly path
  253. private IPlugin myInstance = null;
  254. private string myAssemblyPath = "";
  255. public IPlugin Instance
  256. {
  257. get {return myInstance;}
  258. set {myInstance = value;}
  259. }
  260. public string AssemblyPath
  261. {
  262. get {return myAssemblyPath;}
  263. set {myAssemblyPath = value;}
  264. }
  265. }
  266. }