SharpDevelop插件系统代码阅读笔记

2024-04-23 19:12

本文主要是介绍SharpDevelop插件系统代码阅读笔记,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

SharpDevelop插件系统代码阅读笔记

1.插件系统实现细节


/// <summary>
///AddIn.cs 通过在插件dll里面查找类名,再创建对象,这个功能MEF已经可以实现了
/// </summary>
/// <param name="className"></param>
/// <returns></returns>
public object CreateObject(string className)
{Type t = FindType(className);if (t != null)return Activator.CreateInstance(t);elsereturn null;
}

该方法的缺点就是需要传入类名,那么就需要在xml文件里面显示的将类名标记出来,再去解析。但是MEF功能通过Attribute就可以实现,不需要在xml里面显示的标记出来。但是,AddIn的组织方式肯定还是有其独到之处,继续往下看。
那么,获取到对象之后,是怎么组织管理起来的呢,是怎么插入到树节点中去的呢?

2.菜单栏的实现

static object CreateMenuItemFromDescriptor(MenuItemDescriptor descriptor)
{Codon codon = descriptor.Codon;string type = codon.Properties.Contains("type") ? codon.Properties["type"] : "Command";bool createCommand = codon.Properties["loadclasslazy"] == "false";switch (type) {case "Separator":return new MenuSeparator(codon, descriptor.Parameter, descriptor.Conditions);case "CheckBox":return new MenuCheckBox(codon, descriptor.Parameter, descriptor.Conditions);case "Item":case "Command":return new MenuCommand(codon, descriptor.Parameter, descriptor.Conditions);case "Menu":return new Menu(codon, descriptor.Parameter, ConvertSubItems(descriptor.SubItems), descriptor.Conditions);case "Builder":return codon.AddIn.CreateObject(codon.Properties["class"]);default:throw new System.NotSupportedException("unsupported menu item type : " + type);}
}

上面这段代码在MenuService.cs里面定义,菜单栏是AddIn应用的地方之一。
思考:如何结合MEF和AddIn框架在一起?
我的现有代码里面的菜单栏是通过大量重写基类实现的,这样做对后续的扩展并不友好。因此,还是想能不能将插件系统引入到我的项目中来。虽然在反射部分有些许区别,但是这点区别可以先忽略不计。
主要功能:
1.xml文件的结构和解析
2.

菜单栏命令所在文件
SharpDevelop\src\Main\Base\Project\Src\Commands\FileCommands.cs

public class OpenFile : AbstractMenuCommand
{public override void Run(){using (OpenFileDialog fdiag  = new OpenFileDialog()) {fdiag.AddExtension    = true;fdiag.Filter = ProjectService.GetAllFilesFilter();fdiag.FilterIndex     = 0;fdiag.Multiselect     = true;fdiag.CheckFileExists = true;if (fdiag.ShowDialog(SD.WinForms.MainWin32Window) == DialogResult.OK) {OpenFiles(Array.ConvertAll(fdiag.FileNames, FileName.Create));}}}protected virtual void OpenFiles(FileName[] fileNames){foreach (var name in fileNames) {SD.FileService.OpenFile(name);}}
}

如上面代码,继承自 AbstractMenuCommand

2. 项目树

public abstract class AbstractProjectBrowserTreeNode : ExtTreeNode, IDisposable

ProjectBrowserPanel 这个里面应该有项目Pad里面的右键菜单

public ProjectBrowserPanel(){projectBrowserControl      = new ProjectBrowserControl();projectBrowserControl.Dock = DockStyle.Fill;Controls.Add(projectBrowserControl);if (SD.AddInTree.GetTreeNode("/SharpDevelop/Pads/ProjectBrowser/ToolBar/Standard", false) != null) {toolStrip = SD.WinForms.ToolbarService.CreateToolStrip(this, "/SharpDevelop/Pads/ProjectBrowser/ToolBar/Standard");toolStrip.ShowItemToolTips  = true;toolStrip.Dock = DockStyle.Top;toolStrip.GripStyle = System.Windows.Forms.ToolStripGripStyle.Hidden;toolStrip.Stretch   = true;standardItems = new ToolStripItem[toolStrip.Items.Count];toolStrip.Items.CopyTo(standardItems, 0);Controls.Add(toolStrip);}projectBrowserControl.TreeView.BeforeSelect += TreeViewBeforeSelect;}

SD.AddInTree.GetTreeNode()和SD.WinForms.ToolbarService.CreateToolStrip()这两个方法是关键。

WinFormsToolbarService
ToolbarService.CreateToolStrip(object owner, string addInTreePath)
AddInTree.GetTreeNode(addInTreePath)

public static AddInTreeNode GetTreeNode(string path, bool throwOnNotFound = true)
{var addInTree = ServiceSingleton.GetRequiredService<IAddInTree>();return addInTree.GetTreeNode(path, throwOnNotFound);
}public AddInTreeNode GetTreeNode(string path, bool throwOnNotFound = true)
{if (path == null || path.Length == 0) {return rootNode;}string[] splittedPath = path.Split('/');AddInTreeNode curPath = rootNode;for (int i = 0; i < splittedPath.Length; i++) {if (!curPath.ChildNodes.TryGetValue(splittedPath[i], out curPath)) {if (throwOnNotFound)throw new TreePathNotFoundException(path);elsereturn null;}}return curPath;
}//TreeNode是怎么创建的
AddInTreeNode CreatePath(AddInTreeNode localRoot, string path)
{if (path == null || path.Length == 0) {return localRoot;}string[] splittedPath = path.Split('/');AddInTreeNode curPath = localRoot;int i = 0;while (i < splittedPath.Length) {if (!curPath.ChildNodes.ContainsKey(splittedPath[i])) {curPath.ChildNodes[splittedPath[i]] = new AddInTreeNode();//ChildNodes是一个哈希表,key是节点名字,value只是一个实例}curPath = curPath.ChildNodes[splittedPath[i]];++i;}return curPath;
}

3.1 项目窗口

public static void ShowProjectOptions(IProject project)
{if (project == null){return;}foreach (IViewContent viewContent in SD.Workbench.ViewContentCollection){ProjectOptionsView projectOptions = viewContent as ProjectOptionsView;if (projectOptions != null && projectOptions.Project == project){projectOptions.WorkbenchWindow.SelectWindow();return;}}try{AddInTreeNode projectOptionsNode = AddInTree.GetTreeNode("/SharpDevelop/BackendBindings/ProjectOptions/" + project.Language);ProjectOptionsView projectOptions = new ProjectOptionsView(projectOptionsNode, project);SD.Workbench.ShowView(projectOptions);}catch (TreePathNotFoundException){MessageService.ShowError("${res:Dialog.ProjectOptions.NoPanelsInstalledForProject}");}
}

项目树的解析

public ProjectBrowserPanel()
{projectBrowserControl      = new ProjectBrowserControl();projectBrowserControl.Dock = DockStyle.Fill;Controls.Add(projectBrowserControl);if (SD.AddInTree.GetTreeNode("/SharpDevelop/Pads/ProjectBrowser/ToolBar/Standard", false) != null) {toolStrip = SD.WinForms.ToolbarService.CreateToolStrip(this, "/SharpDevelop/Pads/ProjectBrowser/ToolBar/Standard");toolStrip.ShowItemToolTips  = true;toolStrip.Dock = DockStyle.Top;toolStrip.GripStyle = System.Windows.Forms.ToolStripGripStyle.Hidden;toolStrip.Stretch   = true;standardItems = new ToolStripItem[toolStrip.Items.Count];toolStrip.Items.CopyTo(standardItems, 0);Controls.Add(toolStrip);}projectBrowserControl.TreeView.BeforeSelect += TreeViewBeforeSelect;
}
```cpp
/// <summary>
/// 查看解决方案
/// </summary>
/// <param name="solution"></param>
public void ViewSolution(ISolution solution)
{AbstractProjectBrowserTreeNode solutionNode = new SolutionNode(solution);treeView.Clear();solutionNode.AddTo(treeView);foreach (var treeObject in solution.Items)//Items里面是数据{if (treeObject is IProject){NodeBuilders.AddProjectNode(solutionNode, (IProject)treeObject);//如果是Project,则加入到Builders里面}else{SolutionFolderNode folderNode = new SolutionFolderNode((ISolutionFolder)treeObject);//文件夹节点folderNode.InsertSorted(solutionNode);//讲自己插进父亲节点中}}solutionNode.Expand();//展开
}

4 系统里面那么多右键菜单怎么实现

public class ProjectBrowserPanel : System.Windows.Forms.UserControl
{
ToolStrip toolStrip;//工具栏
ProjectBrowserControl projectBrowserControl;//项目浏览器控件

/// <summary>
/// 从一个解决方案(Solution)中构建一个树视图
/// </summary>
/// <param name="solution"></param>
public void ViewSolution(ISolution solution)
{AbstractProjectBrowserTreeNode solutionNode = new SolutionNode(solution);treeView.Clear();solutionNode.AddTo(treeView);//创建一个解决方案节点(solutionNode),并将其添加到树视图中。foreach (var treeObject in solution.Items)//Items里面是数据,遍历解决方案中的每个项目和服务器{if (treeObject is IProject){NodeBuilders.AddProjectNode(solutionNode, (IProject)treeObject);//对于每个项目,创建一个项目节点(projectNode)并将其添加到项目节点中。}else{SolutionFolderNode folderNode = new SolutionFolderNode((ISolutionFolder)treeObject);//对于每个文件夹,创建一个文件夹节点(folderNode)。folderNode.InsertSorted(solutionNode);//并将folderNode插入到父节点中}}solutionNode.Expand();//展开解决方案节点以显示树视图。
}

这篇关于SharpDevelop插件系统代码阅读笔记的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/929636

相关文章

Django开发时如何避免频繁发送短信验证码(python图文代码)

《Django开发时如何避免频繁发送短信验证码(python图文代码)》Django开发时,为防止频繁发送验证码,后端需用Redis限制请求频率,结合管道技术提升效率,通过生产者消费者模式解耦业务逻辑... 目录避免频繁发送 验证码1. www.chinasem.cn避免频繁发送 验证码逻辑分析2. 避免频繁

精选20个好玩又实用的的Python实战项目(有图文代码)

《精选20个好玩又实用的的Python实战项目(有图文代码)》文章介绍了20个实用Python项目,涵盖游戏开发、工具应用、图像处理、机器学习等,使用Tkinter、PIL、OpenCV、Kivy等库... 目录① 猜字游戏② 闹钟③ 骰子模拟器④ 二维码⑤ 语言检测⑥ 加密和解密⑦ URL缩短⑧ 音乐播放

Python使用Tenacity一行代码实现自动重试详解

《Python使用Tenacity一行代码实现自动重试详解》tenacity是一个专为Python设计的通用重试库,它的核心理念就是用简单、清晰的方式,为任何可能失败的操作添加重试能力,下面我们就来看... 目录一切始于一个简单的 API 调用Tenacity 入门:一行代码实现优雅重试精细控制:让重试按我

Linux系统中查询JDK安装目录的几种常用方法

《Linux系统中查询JDK安装目录的几种常用方法》:本文主要介绍Linux系统中查询JDK安装目录的几种常用方法,方法分别是通过update-alternatives、Java命令、环境变量及目... 目录方法 1:通过update-alternatives查询(推荐)方法 2:检查所有已安装的 JDK方

Linux系统之lvcreate命令使用解读

《Linux系统之lvcreate命令使用解读》lvcreate是LVM中创建逻辑卷的核心命令,支持线性、条带化、RAID、镜像、快照、瘦池和缓存池等多种类型,实现灵活存储资源管理,需注意空间分配、R... 目录lvcreate命令详解一、命令概述二、语法格式三、核心功能四、选项详解五、使用示例1. 创建逻

Python实现MQTT通信的示例代码

《Python实现MQTT通信的示例代码》本文主要介绍了Python实现MQTT通信的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录1. 安装paho-mqtt库‌2. 搭建MQTT代理服务器(Broker)‌‌3. pytho

Spring Boot Maven 插件如何构建可执行 JAR 的核心配置

《SpringBootMaven插件如何构建可执行JAR的核心配置》SpringBoot核心Maven插件,用于生成可执行JAR/WAR,内置服务器简化部署,支持热部署、多环境配置及依赖管理... 目录前言一、插件的核心功能与目标1.1 插件的定位1.2 插件的 Goals(目标)1.3 插件定位1.4 核

MySQL进行数据库审计的详细步骤和示例代码

《MySQL进行数据库审计的详细步骤和示例代码》数据库审计通过触发器、内置功能及第三方工具记录和监控数据库活动,确保安全、完整与合规,Java代码实现自动化日志记录,整合分析系统提升监控效率,本文给大... 目录一、数据库审计的基本概念二、使用触发器进行数据库审计1. 创建审计表2. 创建触发器三、Java

使用Python构建一个高效的日志处理系统

《使用Python构建一个高效的日志处理系统》这篇文章主要为大家详细讲解了如何使用Python开发一个专业的日志分析工具,能够自动化处理、分析和可视化各类日志文件,大幅提升运维效率,需要的可以了解下... 目录环境准备工具功能概述完整代码实现代码深度解析1. 类设计与初始化2. 日志解析核心逻辑3. 文件处

golang程序打包成脚本部署到Linux系统方式

《golang程序打包成脚本部署到Linux系统方式》Golang程序通过本地编译(设置GOOS为linux生成无后缀二进制文件),上传至Linux服务器后赋权执行,使用nohup命令实现后台运行,完... 目录本地编译golang程序上传Golang二进制文件到linux服务器总结本地编译Golang程序