コンポーネント (System.ComponentModel.Component) を使用しており、プロジェクト内にファイルを作成するためにプロジェクトのアプリケーション パスを取得したいと考えています。
どうも
フロリアン
コンポーネント (System.ComponentModel.Component) を使用しており、プロジェクト内にファイルを作成するためにプロジェクトのアプリケーション パスを取得したいと考えています。
どうも
フロリアン
私にとって(一貫して)機能しているように見えた唯一のことは、(EditValue()から取得したIServiceProviderから)EnvDTE.DTEを取得することでした。
EnvDTE.DTE dte = envProvider.GetService(typeof(EnvDTE.DTE)) as EnvDTE.DTE;
string dir = Path.GetDirectoryName(dte.Solution.FullName);
Assembly.GetXAssembly を使用しようとすると、VS が設計時に使用する一時パスを取得しました。
string solutionpath = Directory.GetParent(Application.ExecutablePath).Parent.Parent.Parent.FullName;
「System.IOの使用」以外のライブラリを追加する必要がないため、これが最善の解決策だと思います:)
ロバートはほぼ正しいと思います。
これはうまくいくようです:
Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location)
見てみましょう:コンパイル時のプロジェクト (ビン) フォルダー パス?
この手法を使用すると、「ビルド」(1 回) した後、生成されたファイルを使用して、DesignTime でプロジェクトの場所を取得できます。これは、ワークスペースに依存せずに機能します。
これは機能しますか?
New Uri(Assembly.GetCallingAssembly().CodeBase).AbsolutePath
(実行パスを取得するメソッドをサービス層(アセンブリ)に入れることができるので「CallingAssembly」)
AppDomain.CurrentDomain.BaseDirectory を使用します。
WPFデザイナーについて話している場合は、「コンテキスト」プロパティ/タイプを使用してください
詳細:- 設計時に、modelItem のインスタンスがあります (私はそれを知っていると思います)。そうでない場合は、Activate メソッドの Override 実装でインスタンス化できます。
// DesignAdorner クラス内
public class DesignAdorner : PrimarySelectionAdornerProvider
{
protected override void Activate(ModelItem item)
{
modelItem = item;
}
}
次の単一行コードを使用して、現在のアプリケーション パスにアクセスできるようになりました。
string aplicationPathDir = System.IO.Directory.GetParent(modelItem.Context.ToString()).FullName;
役に立たない場合はお知らせください。
.NetFrameWork 4.5 アセンブリにある envdte80 への参照を追加します。
DTE2 dte = DesignTimeProjectPath.Processes.GetDTE();
if (dte != null)
{
var solution = dte.Solution;
if (solution != null)
{
string baseDir = Path.GetDirectoryName(solution.FullName);
MessageBox.Show(baseDir);
}
}
namespace DesignTimeProjectPath
{
/// <summary>
/// This class takes care of fetching the correct DTE instance for the current process
/// The current implementation works it way down from Visual Studio version 20 to 10 so
/// it should be farely version independent
/// </summary>
public static class Processes
{
[DllImport("ole32.dll")]
private static extern void CreateBindCtx(int reserved, out IBindCtx ppbc);
[DllImport("ole32.dll")]
private static extern void GetRunningObjectTable(int reserved, out IRunningObjectTable prot);
private const int m_MaxVersion = 20;
private const int m_MinVersion = 10;
internal static DTE2 GetDTE()
{
DTE2 dte = null;
for (int version = m_MaxVersion; version >= m_MinVersion; version--)
{
string versionString = string.Format("VisualStudio.DTE.{0}.0", version);
dte = Processes.GetCurrent(versionString);
if (dte != null)
{
return dte;
}
}
throw new Exception(string.Format("Can not get DTE object tried versions {0} through {1}", m_MaxVersion, m_MinVersion));
}
/// <summary>
/// When multiple instances of Visual Studio are running there also multiple DTE available
/// The method below takes care of selecting the right DTE for the current process
/// </summary>
/// <remarks>
/// Found this at: http://stackoverflow.com/questions/4724381/get-the-reference-of-the-dte2-object-in-visual-c-sharp-2010/27057854#27057854
/// </remarks>
private static DTE2 GetCurrent(string versionString)
{
Process parentProc = GetParent(Process.GetCurrentProcess());
int parentProcId = parentProc.Id;
string rotEntry = String.Format("!{0}:{1}", versionString, parentProcId);
IRunningObjectTable rot;
GetRunningObjectTable(0, out rot);
IEnumMoniker enumMoniker;
rot.EnumRunning(out enumMoniker);
enumMoniker.Reset();
IntPtr fetched = IntPtr.Zero;
IMoniker[] moniker = new IMoniker[1];
while (enumMoniker.Next(1, moniker, fetched) == 0)
{
IBindCtx bindCtx;
CreateBindCtx(0, out bindCtx);
string displayName;
moniker[0].GetDisplayName(bindCtx, null, out displayName);
if (displayName == rotEntry)
{
object comObject;
rot.GetObject(moniker[0], out comObject);
return (EnvDTE80.DTE2)comObject;
}
}
return null;
}
private static Process GetParent(Process process)
{
var processName = process.ProcessName;
var nbrOfProcessWithThisName = Process.GetProcessesByName(processName).Length;
for (var index = 0; index < nbrOfProcessWithThisName; index++)
{
var processIndexdName = index == 0 ? processName : processName + "#" + index;
var processId = new PerformanceCounter("Process", "ID Process", processIndexdName);
if ((int)processId.NextValue() == process.Id)
{
var parentId = new PerformanceCounter("Process", "Creating Process ID", processIndexdName);
return Process.GetProcessById((int)parentId.NextValue());
}
}
return null;
}
}
}