在 Visual C# 2010 中获取 DTE2 对象的引用
我想在 Visual Studio 2010 中使用带有 C# 的 DTE2 对象来获取对当前解决方案的引用。
我首先尝试了以下代码:
var dte = Marshal.GetActiveObject("VisualStudio.DTE.10.0") as EnvDTE80.DTE2;
但是当我打开 2 个解决方案,并且此代码位于第一个解决方案中时,我没有得到引用当前的解决方案,而是对我加载的最后一个解决方案的引用。我需要当前的解决方案...
在互联网上搜索,我在 如何从 VSPackage 获取当前解决方案目录?:
// Get an instance of the currently running Visual Studio IDE
DTE dte = (DTE)GetService(typeof(DTE));
但是当我使用它时,我的 dte 对象始终为 NULL。
那么如何在 .net Framework 4.0 上使用 C# 获取 VS2010 中当前的解决方案对象呢?
I want to get a reference to the current solution, using the DTE2 object with C# in Visual Studio 2010.
I first tried the following code:
var dte = Marshal.GetActiveObject("VisualStudio.DTE.10.0") as EnvDTE80.DTE2;
But when I open 2 solutions, and this code is in the first solution, I get NOT a reference to the current solution, but a reference to the last solution I loaded. I need the current solution...
Searching on the internet, I found the following solution in How do you get the current solution directory from a VSPackage?:
// Get an instance of the currently running Visual Studio IDE
DTE dte = (DTE)GetService(typeof(DTE));
But when I use this, my dte object is always NULL.
So how do I get to my current solution object in VS2010 using C# on .net framework 4.0?
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(5)
我觉得以下几点令人不安,所以我解决了它们并找到了适合我的解决方案:
GetActiveObject("VisualStudio.DTE.10.0")
仅适用于第一个打开的(我想)Visual Studio内部静态 DTE2 GetCurrent()
方法需要 Visual Studio 进程 ID。如果您从加载项运行代码(我认为),那很好,但在单元测试中不起作用。我还从 GetCurrent 方法开始,取自 此处。问题是,我不知道如何获取正确的 VisualStudio 进程的 ProcessId(通常有多个实例正在运行)。因此,我采取的方法是获取所有 VisualStudio ROT 条目及其 DTE2,然后将 DTE2.Solution.FullName 与执行程序集位置进行比较(您是否看到更好的选择?)。虽然我承认这不是非常精确的科学,但如果您没有相当特殊的输出路径配置,它应该可以工作。
然后我发现在调试模式下运行代码并访问 DTE2 COM 对象会引发以下异常: System.Runtime.InteropServices.COMException:调用被被调用者拒绝。 (HRESULT 异常:0x80010001 (RPC_E_CALL_REJECTED))。不过,有一种补救措施,称为 MessageFilter< /a>.为了完整起见,我已将代码放在底部。
包含测试方法(使用示例)、调整后的 GetCurrent 方法和用于字符串比较的辅助方法的测试类:
MessageFilter 类:
I felt the following points are unnerving, so I have addressed them and found a solution that works for me:
GetActiveObject("VisualStudio.DTE.10.0")
only works for the first opened (I suppose) Visual Studiointernal static DTE2 GetCurrent()
method of Dennis' answer needs the Visual Studio process Id. That is fine if you run the code from add-ins (I think), but does not work e.g. in unit tests.I also started out with the GetCurrent method, taken from here. Problem was, I didn't know how to get to the ProcessId of the right VisualStudio process (usually multiple instances are running). So the approach I took was getting all VisualStudio ROT entries and their DTE2, then comparing DTE2.Solution.FullName to the executing assembly location (do you see a better choice?). While I readily admit this is not very exact science, it should work if you do not have rather special Output Path configurations.
Then I found that running my code in debug mode and accessing the DTE2 COM objects threw the following exception:
System.Runtime.InteropServices.COMException: Call was rejected by callee. (Exception from HRESULT: 0x80010001 (RPC_E_CALL_REJECTED))
. There is a remedy for that though, called MessageFilter. I have included the code at the bottom for completeness sake.Test class containing a test method (usage example), the adjusted
GetCurrent
method and a helper method for string comparison:MessageFilter class:
我知道这是一个旧线程,但我们需要将此代码与多个 Visual Studio 版本一起使用。我们将代码调整如下:
I know this is an old thread, but we needed to use this code with multiple Visual Studio versions. We tweaked the code as below:
对于任何有兴趣使用 F# 执行此操作的人,这里提供了大部分完整的转换(当前设置为在 linqpad 中运行):
for anyone interested in doing this with F# a mostly complete conversion is here ( currently set to run in linqpad):
我在下面提出了更舒适的完美解决方案(没有火箭科学)。这个方法的工作原理是从 Visual Studio 20 降到 10,以找到独立于 VS 版本的 DTE。
I've made the perfect solution below a bit more comfortabel (no Rocket Science). This one works it's way down from Visual Studio 20 to 10 to find the DTE independent of VS versions.
经过一番广泛的搜索和尝试,我终于使用添加到 MSDN 页面的评论得到了答案:http: //msdn.microsoft.com/en-us/library/ms228755.aspx
我向我的 C# 项目添加了一个静态类:
当我想要访问当前 IDE 时:
但请记住......该代码在调试期间将不起作用!以字符串 rotEntry... 开头的代码行调用 Process.GetCurrentProcess 来获取当前进程的 ID。
在调试我的插件中的某些功能时(使用 MME http://mme.codeplex.com/),我调用了一个需要当前IDE。我使用调用 addin 方法的 ConsoleApp 对此进行测试。在获取当前 IDE 时,当前进程不是 IDE,而是 ConsoleApp.vshost.exe。所以我的代码在调试期间不起作用,但在构建插件并安装此插件后确实起作用。
After some extensive searching and trying i finally got the answer using the comment that was added to the MSDN page: http://msdn.microsoft.com/en-us/library/ms228755.aspx
I added a static class to my c# project:
And at the point that I want to access the current IDE:
But remember.... This code will NOT work during debugging!!! The line of code starting with string rotEntry... has a call to the Process.GetCurrentProcess to get the ID of the current process.
While debugging some functionality in my addin (using MME http://mme.codeplex.com/) I call a method that needs the current IDE. I test this with a ConsoleApp that calls the addin method. At the point of getting the current IDE, the currentprocess is NOT the IDE, but the ConsoleApp.vshost.exe. So my code did not work during debugging, but DID work after building the addin and installing this addin.