亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 開發 > 綜合 > 正文

Management Studio 插件生成安裝包要點(以ProjkyAddin為例)

2024-07-21 02:49:05
字體:
來源:轉載
供稿:網友
Management Studio 插件生成安裝包要點(以PRojkyAddin為例)

通過Visual Studio向導生成Management 插件框架就不說了,網上能搜到不少資料。本篇重點是說明怎么設計一個插件安裝包,適用于Management Studio 2005 到2014的版本。

先講明這么做要面臨的幾個難點:

1、SSMS 2008 和 SSMS 2008 R2的安裝包注冊表項名稱一樣,但只能放一個。也就是,如果只放SSMS 2008的注冊表項,SSMS 2008 R2 啟動的時候會嘗試讀取,并報錯,錯誤本質上是SSMS 2008 R2程序集和SSMS 2008 不同 。同理,如果只是單獨放SSMS 2008 R2的注冊表項,在SSMS 2008的機器上也會報錯。

2、SSMS 2012 和 SSMS 2014查找插件的方法不再通過注冊表了,而是通過目錄上特定的文件來查找的,這個是個關鍵,如果你不知道該把向導生成的*.addin放哪兒,SSMS 就不能正常加載。

3、要同時兼容.NET Framework 2.0 - 4.0 并具有Windows Installer安裝項目的只有Visual Studio 2010,必須采用VS 2010來編寫。

下面,進入正題。

先拿最簡單的SSMS 2005來說,對應于ProjkyAddin項目的ForYukon項目。

項目ForYukon中,編譯后程序集文件名稱是“Projky.ForYukon.dll”,插件加載時必須指定一個實現“IDTExtensibility2”的類,這在向導中就已經生成好了的,這里是“Projky.ForYukon.Connect”類。

那么,接下來就是讓“Projky.ForYukon”插件項目,在SSMS 2005中正常加載,步驟如下。

同一解決方案中,添加一個名為“Projky.Setup”的安裝項目,在Setup項目上右鍵-〉“添加”-〉“項目輸出”,從下拉列表中選擇“Projky.ForYukon”,然后會看到Setup項目下包含一“主輸出來自Projky.ForYukon(活動)”的項。到這里還不夠,選中該主輸出項,右鍵-〉“屬性”,將“Register”項設置為“vsdrpCOM”。最關鍵的是下一步,設置注冊表項。在Setup項目上,右鍵-〉“視圖”-〉“注冊表”。經過本人多次測試,依下圖的方式設置,兼容性最好。

特別說明的是,注冊項位于LocalMachine根下,里面的“Addins”下項“Projky.ForYukon.Connect”是“Projky.ForYukon.dll”中實現了“IDTExtensibility2”接口的類全名。

如果只是單獨需要SSMS 2008 或 2008R2中的一個插件的話,可依上一步的操作即可。我們的目標是讓它們同時有效,而且加載時互不干擾。

針對SSMS 2008的插件項目名稱是“Projky.ForKatmai”,SSMS 2008R2的插件項目名稱是“Projky.ForKilimanjaro”,這里我們的訣竅是采用一個插件中間項目“Projky.ForSQL08Transfer”,在Setup的項目中注冊表項里面只保留Transfer的注冊表項。然后,在Transfer項目中同時引用“ForKatmai”和“ForKilimanjaro”項目,在Transfer插件啟動過程中,根據SSMS版本,決定實例化那個一個具體的“IDTExtensibility2”實例。關鍵代碼如下:

using System;using Extensibility;using EnvDTE;using EnvDTE80;namespace Projky.ForSQL08Transfer {    public class Connect : IDTExtensibility2, IDTCommandTarget {        public Connect() {        }        public void OnConnection(object application, ext_ConnectMode connectMode, object addInInst, ref Array custom) {            _addInInstance = (AddIn)addInInst;            _applicationObject = (DTE2)_addInInstance.DTE;            if (_applicationObject.Version.StartsWith("2007")) {                _transfer = new Projky.ForKatmai.Connect();            } else {                _transfer = new Projky.ForKilimanjaro.Connect();            }            _transfer.OnConnection(application, connectMode, addInInst, ref custom);        }        public void OnDisconnection(ext_DisconnectMode disconnectMode, ref Array custom) {            if (_transfer != null) {                _transfer.OnDisconnection(disconnectMode, ref custom);            }        }        public void OnAddInsUpdate(ref Array custom) {            if (_transfer != null) {                _transfer.OnAddInsUpdate(ref custom);            }        }        public void OnStartupComplete(ref Array custom) {            if (_transfer != null) {                _transfer.OnStartupComplete(ref custom);            }        }        public void OnBeginShutdown(ref Array custom) {            if (_transfer != null) {                _transfer.OnBeginShutdown(ref custom);            }        }        public void Exec(string CmdName, vsCommandExecOption ExecuteOption, ref object VariantIn, ref object VariantOut, ref bool Handled) {            var target = _transfer as IDTCommandTarget;            if (target != null) {                target.Exec(CmdName, ExecuteOption, ref VariantIn, ref VariantOut, ref Handled);            }        }        public void QueryStatus(string CmdName, vsCommandStatusTextWanted NeededText, ref vsCommandStatus StatusOption, ref object CommandText) {            var target = _transfer as IDTCommandTarget;            if (target != null) {                target.QueryStatus(CmdName, NeededText, ref StatusOption, ref CommandText);            }        }        DTE2 _applicationObject;        AddIn _addInInstance;        IDTExtensibility2 _transfer = null;    }}
View Code

重點是根據SSMS版本號實例化一個“IDTExtensibility2”對象,再將接口中各項調用轉發到_transfer的對應實現上。

設置后效果圖:

特別說明,對于“Transfer”、“ForKatmai”、“ForKilimanjaro”項目,在Setup項目中都要添加主輸出,并在各自的主輸出項上右鍵-〉屬性,設置“Register”項為“vsdrpCOM”。SSMS 2012 和 SSMS 2014也一樣,就不強調了。

解決了第一個問題和第三個問題,只剩下第二個問題了,就是實現SSMS 2012 和SSMS 2014插件的加載。

這個方法是在“RedGate”插件上應用的,就是將向導生成的附帶Addin結尾的文件放到特定目錄。SSMS 2012放到Environment.GetFolderPath(Environment.SpecialFolder.CommonApplicationData) + @"/Microsoft/SQL Server Management Studio/11.0/Addins"目錄下,SSMS 2014放到Environment.GetFolderPath(Environment.SpecialFolder.CommonApplicationData) + @"/Microsoft/SQL Server Management Studio/12.0/Addins"目錄下。

然而,這不是僅僅復制這兩個文件到對應目錄就完成的事情。通常Addin文件內容如下:

<?xml version="1.0" encoding="gb2312" standalone="no"?><Extensibility xmlns="http://schemas.microsoft.com/AutomationExtensibility">    <HostApplication>        <Name>Microsoft SQL Server Management Studio</Name>        <Version>*</Version>    </HostApplication>    <Addin>        <FriendlyName>ProjkyAddin</FriendlyName>        <Description>ProjkyAddin for sql 2012</Description>        <Assembly>C:/Program Files (x86)/ProjkyAddin/Projky.ForDenali.dll</Assembly>        <FullClassName>Projky.ForDenali.Connect</FullClassName>        <LoadBehavior>1</LoadBehavior>        <CommandPreload>1</CommandPreload>        <CommandLineSafe>0</CommandLineSafe>    </Addin></Extensibility>

注意了,里面Assembly中是包含了路徑的,如果用戶在Setup包安裝過程中選擇了其它目錄,不是默認目錄,那就導致不能正常加載。

綜上,需要一個定制化的安裝過程。在.NET 里面就是實現System.Configuration.Install.Installer類(需要引用System.Configuration程序集),再在Setup項目中指定自定義動作,指向該類。

ProjkyAddin中的實現代碼如下:

using System;using System.Collections;using System.Collections.Generic;using System.ComponentModel;using System.Configuration.Install;using System.IO;using System.Text;namespace Projky.InstallHelper {    [RunInstaller(true)]    public partial class SetupAddinFile : System.Configuration.Install.Installer {        #region Installer        public SetupAddinFile() {            InitializeComponent();        }        protected override void OnAfterUninstall(IDictionary savedState) {            RemoveDenaliAddinFile();            RemoveHekatonAddinFile();            base.OnAfterUninstall(savedState);        }        public override void Rollback(IDictionary savedState) {            RemoveDenaliAddinFile();            RemoveHekatonAddinFile();            base.Rollback(savedState);        }        public override void Install(IDictionary stateSaver) {            base.Install(stateSaver);            SetupDenaliAddinFile();            SetupHekatonAddinFile();            CreateProjkyAddinFolder();            CreateScriptFoldersTextFile();        }        #endregion        void CreateScriptFoldersTextFile() {            string filePath =
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲视频网站在线观看| 国产精品第三页| 精品人伦一区二区三区蜜桃免费| 国产视频欧美视频| 在线丨暗呦小u女国产精品| 亚洲网站视频福利| 国产精品欧美激情| 亚洲欧美激情四射在线日| 中文字幕欧美亚洲| 在线视频欧美日韩| 精品香蕉一区二区三区| 国产日韩欧美电影在线观看| 国产丝袜精品第一页| 68精品久久久久久欧美| 亚洲免费电影一区| 亚洲欧美日韩国产成人| 亚洲精品成人久久| 欧美人在线视频| 亚洲美女自拍视频| 亚洲大胆人体在线| 国产香蕉精品视频一区二区三区| 欧美高清视频免费观看| 91久久综合亚洲鲁鲁五月天| 国产精品久久久久久久久免费看| 欧美精品一区三区| 亚洲欧美中文另类| 国产成人精品视频在线| 国产美女精品视频免费观看| 狠狠躁18三区二区一区| 俺去了亚洲欧美日韩| 国产成人涩涩涩视频在线观看| 狠狠干狠狠久久| 日韩网站在线观看| 亚洲国产一区二区三区在线观看| 欧美电影免费观看高清完整| 日韩大陆欧美高清视频区| 国产精品www色诱视频| 欧美成人黄色小视频| 激情成人中文字幕| 国产精品444| 精品高清美女精品国产区| 久久精品99无色码中文字幕| 色诱女教师一区二区三区| 97在线免费观看视频| 国内精品国产三级国产在线专| 国产免费一区二区三区在线能观看| 久久久这里只有精品视频| 91视频国产精品| 欧美自拍视频在线| 最近2019中文字幕第三页视频| 亚洲高清av在线| 一本一本久久a久久精品综合小说| 26uuu另类亚洲欧美日本老年| 日韩精品免费一线在线观看| 久久综合伊人77777| 欧美福利在线观看| 福利一区视频在线观看| 久久亚洲精品毛片| 久久99热这里只有精品国产| 成人午夜激情免费视频| 欧美第一淫aaasss性| 欧美日韩综合视频网址| 亚洲在线观看视频| 国产精品美女网站| 欧美激情视频在线免费观看 欧美视频免费一| 91久久国产婷婷一区二区| 一本色道久久88综合亚洲精品ⅰ| 国内精品一区二区三区四区| 国产精品视频导航| 欧美国产亚洲精品久久久8v| 国产精品露脸自拍| 性色av一区二区三区在线观看| 日韩精品免费在线视频观看| 97免费视频在线播放| 国产日韩欧美夫妻视频在线观看| 国产精品综合网站| 国产一区私人高清影院| 国产精品v片在线观看不卡| 91美女片黄在线观| 搡老女人一区二区三区视频tv| 欧美一级片免费在线| 日本久久亚洲电影| 久国内精品在线| 亚洲的天堂在线中文字幕| 国产精品激情av在线播放| 日本高清不卡的在线| 欧美激情videos| 欧美激情综合色综合啪啪五月| 亚洲图片欧美午夜| 亚洲国产美女久久久久| 68精品久久久久久欧美| 91亚洲精品久久久| 久久伊人91精品综合网站| 色婷婷久久一区二区| 26uuu另类亚洲欧美日本老年| 欧美性xxxxhd| 欧美限制级电影在线观看| 性欧美xxxx交| 国产精品亚洲网站| 欧美国产日韩一区二区在线观看| 久久综合网hezyo| 国产精品视频区1| 日韩av电影中文字幕| 日韩电视剧在线观看免费网站| 成人激情免费在线| 国产精品久久久久久久久久东京| 欧洲成人性视频| 欧美激情性做爰免费视频| 日本乱人伦a精品| 日韩在线视频网站| 精品无人国产偷自产在线| 亚洲第一网站男人都懂| 欧美怡红院视频一区二区三区| 精品少妇一区二区30p| 欧美日韩xxx| 国产精品久久久久久久av大片| 国产成人精品日本亚洲| 最近2019年中文视频免费在线观看| 亚洲国产日韩欧美在线动漫| www.日韩欧美| 亚洲天堂视频在线观看| 国产精品中文字幕在线| 川上优av一区二区线观看| 91地址最新发布| 久久人人爽人人爽爽久久| 久久深夜福利免费观看| 97色在线观看免费视频| 亚洲免费人成在线视频观看| 久久夜精品香蕉| 久久久久久久999| 国内精品久久久久久久| 亚洲精品一区久久久久久| 国产suv精品一区二区三区88区| 中文字幕精品网| 国产精品久久久久久久久| 欧美高清在线视频观看不卡| 久久久久久久久网站| 国产不卡精品视男人的天堂| 亚洲图片在线综合| 精品视频在线观看日韩| 日韩中文字幕免费看| 91精品久久久久久久久久入口| 亚洲视频自拍偷拍| 欧美巨乳美女视频| 欧美中文字幕视频在线观看| 国产精品一区二区3区| 欧美激情一二三| 久久久久久国产精品三级玉女聊斋| 日韩电影在线观看中文字幕| 欧美极品美女电影一区| 欧美精品videosex牲欧美| 亚洲欧美在线播放| 久久精品色欧美aⅴ一区二区| 国产精品国产三级国产专播精品人| 亚洲国产精品va在线看黑人动漫| 蜜月aⅴ免费一区二区三区| 国产精品久久久久久久7电影| 国产精品久久久久久久久久三级| 亚洲欧美激情视频| 综合av色偷偷网| 久久天堂av综合合色| 国产精品av免费在线观看| 91精品久久久久久久久青青| 日韩精品在线免费|