VS扩展开发框架

VsSharp:一个VS扩展开发框架(上)

上篇:设计

一、引子

自2008年起开发SSMS插件SqlSharp(er)的过程中,有一天发现多数代码都大同小异,就像这样。

 Commands2 commands = (Commands2)_applicationObject.Commands;
                string toolsMenuName = "Tools";

                //Place the command on the tools menu.
                //Find the MenuBar command bar, which is the top-level command bar holding all the main menu items:
                Microsoft.VisualStudio.CommandBars.CommandBar menuBarCommandBar = ((Microsoft.VisualStudio.CommandBars.CommandBars)_applicationObject.CommandBars)["MenuBar"];

                //Find the Tools command bar on the MenuBar command bar:
                CommandBarControl toolsControl = menuBarCommandBar.Controls[toolsMenuName];
                CommandBarPopup toolsPopup = (CommandBarPopup)toolsControl;

                //This try/catch block can be duplicated if you wish to add multiple commands to be handled by your Add-in,
                //  just make sure you also update the QueryStatus/Exec method to include the new command names.
                try
                {
                    //Add a command to the Commands collection:
                    // add + (int)vsCommandStatus.vsCommandStatusEnabled if we want the default state to be enabled
                    Command command = commands.AddNamedCommand2(_addInInstance, "FormatSQL", "Format SQL", "Format SQL", true, 59, ref contextGUIDS, (int)vsCommandStatus.vsCommandStatusSupported + (int)vsCommandStatus.vsCommandStatusEnabled, (int)vsCommandStyle.vsCommandStylePictAndText, vsCommandControlType.vsCommandControlTypeButton);

                    //Add a control for the command to the tools menu:
                    if ((command != null) && (toolsPopup != null))
                    {
                        command.AddControl(toolsPopup.CommandBar, 1);
                    }
                }
                catch (System.ArgumentException)
                {
                    //If we are here, then the exception is probably because a command with that name
                    //  already exists. If so there is no need to recreate the command and we can
                    //  safely ignore the exception.
                }

于是萌生出开发一个框架的想法。

于是有了一个叫SsmsSharp的框架,后正式命名为SqlSharp发布到了CodePlex上。

与此同时,将操纵EnvDTE的代码与SSMS Objects的代码分离,操纵EnvDTE的代码就形成了本篇要说的VsSharp。

后来,当我正式使用VsSharp开发VS扩展时,又引出一些新问题如源代码签出、一个VS搭载多个扩展,解决这些问题后,VsSharp开始成熟起来。

二、设计思路

1、目标

应用Command模式,定义每个控件的行为。将一个个控件的属性与行为集合在一个配置文件中,在VS启动时自动加载控件,点击控件时通过反射触发相应的命令。

2、流程

User:终端用户(也就是各位码农)

Host:VS实例,提供全局的EnvDTE对象访问器,注册Plugin,响应IDE的各种事件(如文档打开关闭等)

Plugin:基于VsSharp开发的插件(此处为避免与EnvDTE.AddIn重名,命名为Plugin)

由此引出VsSharp的职责

  1. 负责配置的加载
  2. 向VS注册控件
  3. 响应用户的点击及其他事件

三、概要设计

1、对象设计

1.1 基于上述职责定义,抽象出如下对象:

  • CommandConfig:负责命令与控件的配置描述
  • CommandManager:负责配置的加载,和与VS的交互
  • CommandBarAccessor:与VS直接交互的对象,实现ICommandBarAccessor接口,主要是为了隔离VS版本的差异
  • Host:宿主,单例,表示当前操作的VS实例;CommandAccessor通过它与EnvDTE交互
  • PlugIn:主要属性为CommandConfig和Connect入口的Assembly

CommandBarAccessor的行为:

 public interface ICommandBarAccessor
    {
        void AddControl(CommandControl control);
        void ResetControl(CommandControl control);
        void EnableControls(IEnumerable<string> ids ,bool enabled);
        void Delete();
    }

  • AddContro:添加一个控件
  • ResetControl:重置控件,比如某控件的子菜单可以依赖于特定的配置或数据源,当配置或数据源发生变化时,需要重新加载控件
  • EnableControl:启用(禁用)控件,比如某些控件是用于操作文档的,当有文档打开时才启用
  • Delete:删除控件,当VS退出时执行Disconnect方法时触发

1.2 命令接口

 public interface ICommand
    {
        void Execute(object arg = null);
    }

命令类型:

 public enum CommandActionType
    {
        Menu,
        Program,
        Window,
        Dialog
    }

  • Menu:缺省类型,无任何行为
  • Program:执行一段程序
  • Window:打开一个窗体
  • Dialog:打开一个模态窗体

1.3 命令控件描述

主要有两种控件类型:

  • CommandMenu:包括主菜单栏的自定义菜单、上下文菜单,其下级可以有子菜单
  • CommandButton:主要是插入ToolStrip的ToolStripButton,其下级也可以有子菜单

抽象类CommandControl:CommandMenu和CommandButton的父类,描述控件的ID、文本、图标、命令类型、位置、所属父控件等属性。

以下代码段为CommandControl的全部属性。

其中,

ClassName为供反射用的动作类型名称,当CommandActionType为Program时,要求该类型实现了ICommand接口。

 

CommandMenu继承CommandControl,特有子菜单相关属性。

其中SubMenus属性可在编程时操纵,SubGeneratorType为配置文件中定义的供反射用的子菜单生成器类型,用于启动时根据特定数据源自动生成。

 

2、类图

调用关系 :

  • Connect对象启动时加载CommandConfig,生成一个Plugin对象传给CommandManager,并向Host.Instance注册;CommandManager加载CommandConfig描述的所有控件
  • Connect.OnConnection方法调用CommandManager.Load方法
  • Connect.Exec方法调用CommandManager.Execute方法
  • Connect.OnDisconnection方法调用CommandManager.Disconnect方法

四、源代码

http://vssharp.codeplex.com/

---------------------------------------------------

下篇将以一个实例来讲解框架的使用,敬请期待。

时间: 2024-12-29 18:48:32

VS扩展开发框架的相关文章

VsSharp:一个VS扩展开发框架(上)

上篇:设计 一.引子 自2008年起开发SSMS插件SqlSharp(er)的过程中,有一天发现多数代码都大同小异,就像这样. Commands2 commands = (Commands2)_applicationObject.Commands; string toolsMenuName = "Tools"; //Place the command on the tools menu. //Find the MenuBar command bar, which is the top-

Linux 搭建php扩展开发框架

1.安装phpize(如果是使用php源码编译就免了,本身就有) 2.打开php源码,ext中有ext_skel工具,使用它可以方便 ./ext_skel --extname = myext 生成扩展框架后,需要修改扩展的mp4文件, #cd ext/myext/ #vim config.mp4 修改文件如上.. 3.编译安装扩展 #cd myext #phpize #./configure --with-php-config = /usr/bin/php-config   //php-conf

C语言PHP的helloworld扩展

1.下载php源码 wget http://cn2.php.net/distributions/php-5.6.10.tar.gz tar -zxvf php-5.6.10.tar.gz 2.建立扩展开发框架 ./ext_skel --extname=helloworld cd php-5.6.10/ext/ ./ext_skel --extname=helloworld Creating directory helloworld Creating basic files: config.m4 

VS2012编译php扩展

注意:用VS2015来做会比较好! 开发前准备工作:cygwinvisual studio 2012php编译后的程序      使用的是 xampp集成安装包,所以编译后的程序路径为D:\xampp\phpphp编译前的源码      使用的是 php-5.6.14-src,路径为D:\Hacfin\Code\PHP\php_sdk_win\php-5.6.14-src 1.修改 "ext_skel_win32.php"文件      D:\Hacfin\Code\PHP\php_s

php扩展,一个helloworld的实现

系统环境:ubuntu14.04 php版本:php5.5.9 apache版本:apache2.4 使用的源码包:php 5.6.8 php -v PHP 5.5.9-1ubuntu4.7 (cli) (built: Mar 16 2015 20:47:39) Copyright (c) 1997-2014 The PHP Group Zend Engine v2.5.0, Copyright (c) 1998-2014 Zend Technologies with Zend OPcache

几种流行Webservice框架性能对比

1摘要 开发webservice应用程序中离不开框架的支持,当open-open网站列举的就有很多种,这对于开发者如何选择带来一定的疑惑.性能Webservice的关键要素,不同的框架性能上存在较大差异,而当前在官方网站.网络资料中可以方便的找到各自框架的介绍,但是很少有针对不同框架性能测试数据.本文选择了比较流行几个框架: Apache Axis1.Apache Axis2.Codehaus XFire.Apache CXF.Apache Wink.Jboss  RESTEasy.sun JA

虚拟化三剑客专题-Hyper-V(中)

Hyper-V提供广泛的操作系统支持(包括32/64位OS,例如Windows.Linux等操作系统).虚拟VLAN的支持(对虚拟化环境中的虚拟机划分VLAN,以保证虚拟机间信息的相互隔离).包含了全新的虚拟交换功能(运行Windows网络负载均衡服务,以对不同服务器上的多个虚拟机的负载进行均衡)等. Hyper-V提供了扩展开发框架和API,以便企业能够将自行特有的硬件设备融入到虚拟化平台中,为虚拟机提供虚拟化服务:提供标准的Windows管理架构(WMI)接口以及API接口使得软件供应商和开

DDD开发框架ABP之本地化资源的数据库存储扩展

在上一篇<DDD开发框架ABP之本地化/多语言支持>中,我们知道,ABP开发框架中本地化资源存储可以采用XML文件,RESX资源文件,也提供了其他自定义的存储方式的扩展接口.ABP框架默认实现了前面两种方式,而数据库存储方式则需要自己扩展,大概是因为数据库存储涉及到了实体和仓储等方面的具体内容,不适合放在基本框架里面. 以数据库的方式存储本地化资源,一个最明显的好处就是方便修改,尤其是对于基于数据库的应用系统而言,可以提供统一的维护界面.接下来我们就来一步步地实现将本地化资源存储在数据库中.

基于MVC4+EasyUI的Web开发框架形成之旅--附件上传组件uploadify的使用

很久之前,当我还在用Asp.NET开发一些行业管理系统的时候,就曾经使用这个组件作为文件的上传操作,在随笔<Web开发中的文件上传组件uploadify的使用>中可以看到,Asp.NET中如何使用这个组件进行文件上传的,当时上传文件的处理主要也是使用ashx一般处理程序来进行处理的.本文主要介绍我的Web开发框架中,在MVC4的环境中如何集成这个非常棒的文件上传组件的. 1.上传组件uploadify的说明及脚本引用 Uploadify 是 JQuery 一个著名的上传插件,利用 Flash