I won't go into many explanations about why certain things are done, but all I can say is that there aren't much clear documentation on how to create specific add-ins for SSMS 2008 even though there are for 2005 a lot has changed and some things don't work anymore for 2008 and worse Microsoft have changed some of the interfaces again in 2008 R2 so considering that the dependencies on the API's are flaky which is even more reason to create a flexible plugin with an exploratory environment. With that said I will dive right in and just give the instructions as it is now to get it working in SSMS 2008 R2 using Visual Studio 2010 and .NET 4.0.
Start VS 2010 and Create a new Project
File -> New -> Project
Other Project Types -> Extensibility -> Visual Studio Add-In
Click next to add the default support for hosts.
Ignore the next page's options for now, click next
Lets skip the next page as well:
Click Finish:
The template will be created.
But there is one problem, the default template currently is incorrect and will fail with the latest SSMS open the Connect.cs class and notice the following code:
public void OnConnection(object application, ext_ConnectMode connectMode, object addInInst, ref Array custom) { _applicationObject = (DTE2)application; _addInInstance = (AddIn)addInInst; }
You Have to change this code slightly as follows:
public void OnConnection(object application, ext_ConnectMode connectMode, object addInInst, ref Array custom) { // the application object is no longer the DTE object //_applicationObject = (DTE2)application; _addInInstance = (AddIn)addInInst; // this is the new application object _applicationObject = (DTE2)_addInInstance.DTE; }
The Application object is no longer the DTE application object in the latest SSMS.
Then lets add an interface to our Connect class:
IDTCommandTarget
as follows:
public class Connect : IDTExtensibility2, IDTCommandTarget {
This will allow us to also receive events from any commands that are executed.
So now lets create a controller class, that understands these objects so that this is also more reusable to create future add-ins. where we will handle all these method stubs.
This is the entire class with all the methods we will need:
///The object for implementing an Add-in. public class Connect : IDTExtensibility2, IDTCommandTarget { AddInController _controller; ///Implements the constructor for the Add-in object. Place your initialization code within this method. public Connect() { } ///Implements the OnConnection method of the IDTExtensibility2 interface. Receives notification that the Add-in is being loaded. /// Root object of the host application. /// Describes how the Add-in is being loaded. /// Object representing this Add-in. public void OnConnection(object application, ext_ConnectMode connectMode, object addInInst, ref Array custom) { _controller = new AddInController(addInInst); _controller.OnConnection(application, connectMode, addInInst, ref custom); } ///Implements the OnDisconnection method of the IDTExtensibility2 interface. Receives notification that the Add-in is being unloaded. /// Describes how the Add-in is being unloaded. /// Array of parameters that are host application specific. public void OnDisconnection(ext_DisconnectMode disconnectMode, ref Array custom) { _controller.OnDisconnection(disconnectMode, ref custom); } ///Implements the OnAddInsUpdate method of the IDTExtensibility2 interface. Receives notification when the collection of Add-ins has changed. /// Array of parameters that are host application specific. public void OnAddInsUpdate(ref Array custom) { _controller.OnAddInsUpdate(ref custom); } ///Implements the OnStartupComplete method of the IDTExtensibility2 interface. Receives notification that the host application has completed loading. /// Array of parameters that are host application specific. public void OnStartupComplete(ref Array custom) { _controller.OnStartupComplete(ref custom); } ///Implements the OnBeginShutdown method of the IDTExtensibility2 interface. Receives notification that the host application is being unloaded. /// Array of parameters that are host application specific. public void OnBeginShutdown(ref Array custom) { _controller.OnBeginShutdown(ref custom); } public void Exec(string CmdName, vsCommandExecOption ExecuteOption, ref object VariantIn, ref object VariantOut, ref bool Handled) { _controller.Exec(CmdName, ExecuteOption, ref VariantIn, ref VariantOut, ref Handled); } public void QueryStatus(string CmdName, vsCommandStatusTextWanted NeededText, ref vsCommandStatus StatusOption, ref object CommandText) { _controller.QueryStatus(CmdName, NeededText, ref StatusOption, ref CommandText); } }
Next we would like to create some tool windows, for that I will implement the following function in the controller:
public ToolWindowContext CreateToolWindow(string typeName, string assemblyLocation, string Caption, Guid uiTypeGuid, DTE2 applicationObject, AddIn addinInstance) { Windows2 win2 = applicationObject.Windows as Windows2; if (win2 != null) { object controlObject = null; Window toolWindow = win2.CreateToolWindow2(addinInstance, assemblyLocation, typeName, Caption, "{" + uiTypeGuid.ToString() + "}", ref controlObject); Window2 toolWindow2 = (Window2)toolWindow; toolWindow.Visible = true; return new ToolWindowContext(toolWindow, toolWindow2, controlObject); } return null; }
Next you have to install the add-in, the easiest is to create a deployment package:
Right Click Add -> New Project
Choose Other Project Types -> Setup And Deployment -> Setup Project
Right Click on the newly created project and choose Add -> Project Output
Right Click on the Detected Dependencies and Select Exclude for the Visual Studio DLL's as these are already installed on the machine:
Next, bring up the Registry Editor. Click View -> Editor -> Registry.
Finally set the main output of the project to register for COM:
Locate the Primary Output and view the properties,
Locate the Register Property,
Click the dropdown
Choose vsdrpCOM
Now we are ready to start the basic add, in but first I wanted to create a basic DLL to host some tool windows. I wanted my first tool window to be a log window simply to capture events from my add-in.
What you need to do is place your GUI controls in a seperate DLL. It has to be a .NET 3.5 preferably as it seems that the ide has problems loading a .NET 4 assembly. However I dont need the latest version of .net here for now considering the controls that I plan to create.
So create a new assembly: Add -> New Project -> Class Library
Right click on the new project and Add -> User Control
Name the user Control: UcLogWindow.cs
NOTE: You have to add an attribute to the class to make the container visible to your add in:
[ComVisible(true)]
Add a method to Log a message to the window:
public void LogMessage(string message) { }
The full declaration of the control
public partial class UcLogWindow : UserControl [ComVisible(true)] public partial class UcLogWindow : UserControl { public void LogMessage(string message) { richTextBox1.AppendText(string.Format("[{0:HH:MM:ss}]: {1}{2}", DateTime.Now, message, System.Environment.NewLine); } public UcLogWindow() { InitializeComponent(); } }
Now we will use the CreateToolWindow Function we added earlier to create a log window for our add-in.
IToolWindowContext CreateCommonUIControl(string typeName, string caption, Guid guid) { return _adapter.CreateToolWindow(typeName, _uiDLL, caption, guid); } void CreateLogWindow() { _logWindow = CreateCommonUIControl("ACSR.SqlServer.Addin.Core.UI.UcLogWindow", "IronPython LogWindow", new Guid("3ADC13FF-DCF4-4C49-B2EF-3D78DECDC664")); _logWindowControl = ((dynamic)_logWindow.ControlObject); } public void OnConnection(object application, ext_ConnectMode connectMode, object addInInst, ref Array custom) { _workingDirectory = Path.GetDirectoryName(System.Reflection.Assembly.GetExecutingAssembly().Location); _uiDLL = Path.Combine(_workingDirectory, "ACSR.SqlServer.Addin.Core.UI.dll"); CreateLogWindow(); }
So now we are ready to install the Add-In, Right click on the Setup Project and choose Install:
If you start SSMS you should now see the Add-In Loading with the logwindow.
That's it for preparing the plugin, now we have everything in place to start coding the add-ins.
Our methods are just stubs still and to get them to work and explore all the features of the IDE is big enough for a separate article.
We don't have any IronPython scripting in this Add-In yet, but I have everything I need in place to add script hosting to the next article and Version.
Would really love to know where you got the
ReplyDeleteAddInController _controller;
from ? this is the part that i am getting error
Please read the following blog post after this one:
ReplyDeletehttp://thewalkingdev.blogspot.com/search/label/SQL%20Server%20Add-Ins
This builds on the current post and adds the Add-In using IronPython you can scroll to the bottom to download the code and get the addin controller.
Tks Andre the .msi is working just fine no issues there. But the project itself is not building up as there are some missing dependencies.Possible to send me just the plain simple add-in with out the iron-python included like pop up a message box and such once the add-in is loaded and create a tools menu item like yours tks so much. Great article btw.
ReplyDeletesorry forgot to include my mail
ReplyDeletejhonathan82@gmail.com
can i follow you on twitter or something?.
ReplyDeleteHi, You place your GUI controls in a seperate DLL. If the seperate dll needs to load another assembly(DLL) which is not in GAC, then you will load the seperate DLL failed. How do you resolve this problem? Thanks.
ReplyDeleteHi, You said "What you need to do is place your GUI controls in a seperate DLL. It has to be a .NET 3.5 preferably as it seems that the ide has problems loading a .NET 4 assembly. However I dont need the latest version of .net here for now considering the controls that I plan to create.". Now I have a assembly(DLL) of plugin which is based on .NET 4.0 for SSMS 2008 R2 and a seperate assembly which is based on .NET 4.0. While creating tool window by CreateToolWindow2 I get an System.BadImageFormatException. But it works on SSMS 2012. Are you sure that the IDE has problems loading .Net 4.0 assembly? Thanks.
ReplyDelete