你如何在代码中获得C#(VS 2008)中的解决方案目录?

这里有一个恼人的问题。 我有一个NHibernate / Forms应用程序,我正在通过SVN工作。 我做了一些自己的控件,但是当我拖放那些(或者查看我已经拖放过的一些表单编辑器)到我的其他控件上时,Visual Studio决定它需要执行我编写的一些代码,包括查找hibernate.cfg.xml的部分。

我不知道为什么会这样,但是(有时!)当我在表单加载期间执行代码或拖放它时将当前目录切换到C:\ program files \ vs 9.0 \ common7 \ ide,然后nhibernate抛出一个例外,它无法找到hibernate.cfg.xml,因为我在相对路径中搜索它。

现在,我不想硬编码hibernate.cfg.xml的位置,或者只是将hibernate.cfg.xml复制到ide目录(这将起作用)。 我想要一个获得解决方案目录的解决方案,而当前目录是common7 \ ide。 可以让某人在设计器中查看我的表单,然后在任意机器上的任意目录中进行新的检查。 不,我不打算在代码中加载控件。 我在控件中有如此多的控件,如果没有它,将所有内容排成一行都是一场噩梦。

我尝试了一个预构建事件,它创建了一个包含解决方案目录的文件,但当然如何从common7 \ ide中找到它? 由于svn,所有项目文件都需要位于解决方案目录中。

谢谢你的帮助,我已经花了几个小时摆弄这个徒劳。

更新:我将hibernate.cfg设置为嵌入式资源。 对于每个配置,我只需创建一个新的构建配置,调试,发布,XYZ。 在大多数情况下,我建议嵌入您依赖的任何文件来运行程序。 它使构建安装程序变得更加简单。

这可能有点迟了,但我在http://www.tek-tips.com/viewthread.cfm?qid=1226891&page=164找到了解决方案。 由于我使用的是Visual Studio 2010,因此我进行了一些小的更改。 你必须参考EnvDTE和EnvDTE100(EnvDTE90 for VS2008),

string solutionDirectory = ((EnvDTE.DTE)System.Runtime .InteropServices .Marshal .GetActiveObject("VisualStudio.DTE.10.0")) .Solution .FullName; solutionDirectory = System.IO.Path.GetDirectoryName(solutionDirectory); 

当然我使用的是VisualStudio.DTE.10.0,你应该使用VisualStudio.DTE.9.0。

祝好运!

我终于想出了这个。 这适用于任何Visual Studio版本,不依赖于EnvDTE,并解决了此处提供的原始问题。

  1. 在项目设置中的“构建事件”下,添加以下“预构建事件命令行”:

     echo $(SolutionDir) > ..\..\solutionpath.txt 
  2. 一次构建项目。 该文件将在项目根目录中创建。

  3. 在解决方案资源管理器中,单击“显示所有文件”和“刷新”

  4. 将solutionpath.txt添加到您的解决方案中

  5. 右键单击solutionpath.txt,单击属性。 将构建操作更改为“嵌入式资源”

  6. 使用以下代码获取解决方案路径。

      string assemblyname = System.Reflection.Assembly.GetExecutingAssembly().GetName().Name; string path = ""; using (var stream = System.Reflection.Assembly.GetExecutingAssembly().GetManifestResourceStream(assemblyname + ".solutionpath.txt")) { using (var sr = new StreamReader(stream)) { path = sr.ReadToEnd().Trim(); } } 

因为这是一个预构建事件,所以在构建开始之前不需要存在该文件,因此它与源代码控制兼容并且没有任何明显的问题。

更新 :不幸的是,我不知道如何在设计时获取解决方案文件夹。 所以,从技术上讲,我没有回答你的问题,只提供一个潜在的解决方法。

您可以检查您的控件是否在DesignMode ,如果是,您可以使用Assembly.GetExecutingAssembly()来获取控件的Assembly并确定它的加载位置。

请注意,有一些关于DesignTime属性值的警告,即如果您设计控件或者如果您正在设计包含控件的Form,它将正确返回true,但是如果您正在设计包含包含控件的表单你的控制,它会重新调整错误。

您可能希望跳过整个DesignTime检查,因为如果您找到该配置文件的标准方法失败,请始终在程序集的基本路径中查找NHibernate配置。

听起来你只需要为配置文件编写更好的路径。

如果您这样做:

 configPath = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "\\PathToCFG"); 

当Windows更改您当前目录时,您不应该搞砸。

编辑:您可能遇到Visual Studio托管过程的问题 。 你可以禁用它吗? 项目属性\ debug下有一个复选框。

我最终嵌入了配置文件。 由于程序自动更新独立于配置文件,我可以放弃将其暴露给用户。