我可以在 VSIX 中检测到使用 Visual Studio 工作区保存(未更改)的文档吗?
Can I detect document saved (not changed) with Visual Studio Workspace in a VSIX?
在我的扩展程序中,我使用 DTE 进入文档保存事件:
this._events2 = (Events2)this._dte.Events;
//setup document events
this._documentEvents = this._events2.DocumentEvents;
this._documentEvents.DocumentSaved += _documentEvents_DocumentSaved;
我正在将我的扩展迁移到 VS 2017,并想开始使用 Roslyn 东西而不是 DTE。我想出了如何获取 Visual Studio 工作区并进入 workspace changed event。现在我可以访问所有这些文档事件
/// <summary>
/// A document was added to the current solution.
/// </summary>
DocumentAdded = 9,
/// <summary>
/// A document was removed from the current solution.
/// </summary>
DocumentRemoved = 10,
/// <summary>
/// A document in the current solution was reloaded.
/// </summary>
DocumentReloaded = 11,
/// <summary>
/// A document in the current solution was changed.
/// </summary>
DocumentChanged = 12,
但是没有DocumentSaved。每次击键时 DocumentChange 都会触发,而 DocumentReloaded 似乎根本不会触发。是否可以仅检测使用 roslyn 工作区事件保存的文档?
你有 DTE2 接口而且你有 Events.DocumentEvents.DocumentSaved
我没试过,但看起来很有希望。
要检测文档保存事件( OnBeforeSave() 或 OnAfterSave() ),您可以实现 IVsRunningDocTableEvents3
接口。您可以通过将此接口实现到助手 class 中并公开一个 public 事件 event OnBeforeSaveHandler BeforeSave
和一个 public 委托 delegate void OnBeforeSaveHandler(object sender, Document document)
来做到这一点。
要捕获此事件:runningDocTableEvents.BeforeSave += OnBeforeSave
然后您可以在 OnBeforeSave
方法中编写代码。
我对 IVsRunningDocTableEvents3
接口的实现如下所示:
public class RunningDocTableEvents : IVsRunningDocTableEvents3
{
#region Members
private RunningDocumentTable mRunningDocumentTable;
private DTE mDte;
public delegate void OnBeforeSaveHandler(object sender, Document document);
public event OnBeforeSaveHandler BeforeSave;
#endregion
#region Constructor
public RunningDocTableEvents(Package aPackage)
{
mDte = (DTE)Package.GetGlobalService(typeof(DTE));
mRunningDocumentTable = new RunningDocumentTable(aPackage);
mRunningDocumentTable.Advise(this);
}
#endregion
#region IVsRunningDocTableEvents3 implementation
public int OnAfterAttributeChange(uint docCookie, uint grfAttribs)
{
return VSConstants.S_OK;
}
public int OnAfterAttributeChangeEx(uint docCookie, uint grfAttribs, IVsHierarchy pHierOld, uint itemidOld, string pszMkDocumentOld, IVsHierarchy pHierNew, uint itemidNew, string pszMkDocumentNew)
{
return VSConstants.S_OK;
}
public int OnAfterDocumentWindowHide(uint docCookie, IVsWindowFrame pFrame)
{
return VSConstants.S_OK;
}
public int OnAfterFirstDocumentLock(uint docCookie, uint dwRDTLockType, uint dwReadLocksRemaining, uint dwEditLocksRemaining)
{
return VSConstants.S_OK;
}
public int OnAfterSave(uint docCookie)
{
return VSConstants.S_OK;
}
public int OnBeforeDocumentWindowShow(uint docCookie, int fFirstShow, IVsWindowFrame pFrame)
{
return VSConstants.S_OK;
}
public int OnBeforeLastDocumentUnlock(uint docCookie, uint dwRDTLockType, uint dwReadLocksRemaining, uint dwEditLocksRemaining)
{
return VSConstants.S_OK;
}
public int OnBeforeSave(uint docCookie)
{
if (null == BeforeSave)
return VSConstants.S_OK;
var document = FindDocumentByCookie(docCookie);
if (null == document)
return VSConstants.S_OK;
BeforeSave(this, FindDocumentByCookie(docCookie));
return VSConstants.S_OK;
}
#endregion
#region Private Methods
private Document FindDocumentByCookie(uint docCookie)
{
var documentInfo = mRunningDocumentTable.GetDocumentInfo(docCookie);
return mDte.Documents.Cast<Document>().FirstOrDefault(doc => doc.FullName == documentInfo.Moniker);
}
#endregion
}
当触发来自 VS 的任何类型的保存命令(CTRL + S、全部保存、编译、构建等)时,我使用此实现来格式化一些文档。
如果您想从某个命令(例如 Compile 获取保存事件,您必须在 OnBeforeSave() 方法中添加一个检查添加更多代码
首先,您必须保持对 CommandEvents var vommandEvents = dte.Events.CommandEvents
的强引用,并向 CommandEvents 添加一个新方法 commandEvents.BeforeExecute += CommandEventsBeforeExecute;
.
这会起作用,因为 CommandsEvents 将始终在 BeforeSave 之前调用。这就是 Visual Studio 中的工作原理,每个操作都代表一个需要一些步骤和事件的命令(例如,编译命令在其工作流程中包含保存文档事件)。
public override void OnBeforeSave(object sender, Document aDocument)
{
if (false == myCompileCommandFlag)
return;
// write your code here
}
public void CommandEventsBeforeExecute(string aGuid, int aId, object aCustomIn, object aCustomOut, ref bool aCancelDefault)
{
string commandName = GetCommandName(aGuid, aId);
if (0 != string.Compare("Build.Compile", commandName))
{
return;
}
myCompileCommandFlag= true;
}
public string GetCommandName(string aGuid, int aId)
{
if (null == aGuid)
return string.Empty;
if (null == mCommand)
return string.Empty;
Command cmd = mCommand.Item(aGuid, aId);
if (null == cmd)
return string.Empty;
return cmd.Name;
}
在我的扩展程序中,我使用 DTE 进入文档保存事件:
this._events2 = (Events2)this._dte.Events;
//setup document events
this._documentEvents = this._events2.DocumentEvents;
this._documentEvents.DocumentSaved += _documentEvents_DocumentSaved;
我正在将我的扩展迁移到 VS 2017,并想开始使用 Roslyn 东西而不是 DTE。我想出了如何获取 Visual Studio 工作区并进入 workspace changed event。现在我可以访问所有这些文档事件
/// <summary>
/// A document was added to the current solution.
/// </summary>
DocumentAdded = 9,
/// <summary>
/// A document was removed from the current solution.
/// </summary>
DocumentRemoved = 10,
/// <summary>
/// A document in the current solution was reloaded.
/// </summary>
DocumentReloaded = 11,
/// <summary>
/// A document in the current solution was changed.
/// </summary>
DocumentChanged = 12,
但是没有DocumentSaved。每次击键时 DocumentChange 都会触发,而 DocumentReloaded 似乎根本不会触发。是否可以仅检测使用 roslyn 工作区事件保存的文档?
你有 DTE2 接口而且你有 Events.DocumentEvents.DocumentSaved
我没试过,但看起来很有希望。
要检测文档保存事件( OnBeforeSave() 或 OnAfterSave() ),您可以实现 IVsRunningDocTableEvents3
接口。您可以通过将此接口实现到助手 class 中并公开一个 public 事件 event OnBeforeSaveHandler BeforeSave
和一个 public 委托 delegate void OnBeforeSaveHandler(object sender, Document document)
来做到这一点。
要捕获此事件:runningDocTableEvents.BeforeSave += OnBeforeSave
然后您可以在 OnBeforeSave
方法中编写代码。
我对 IVsRunningDocTableEvents3
接口的实现如下所示:
public class RunningDocTableEvents : IVsRunningDocTableEvents3
{
#region Members
private RunningDocumentTable mRunningDocumentTable;
private DTE mDte;
public delegate void OnBeforeSaveHandler(object sender, Document document);
public event OnBeforeSaveHandler BeforeSave;
#endregion
#region Constructor
public RunningDocTableEvents(Package aPackage)
{
mDte = (DTE)Package.GetGlobalService(typeof(DTE));
mRunningDocumentTable = new RunningDocumentTable(aPackage);
mRunningDocumentTable.Advise(this);
}
#endregion
#region IVsRunningDocTableEvents3 implementation
public int OnAfterAttributeChange(uint docCookie, uint grfAttribs)
{
return VSConstants.S_OK;
}
public int OnAfterAttributeChangeEx(uint docCookie, uint grfAttribs, IVsHierarchy pHierOld, uint itemidOld, string pszMkDocumentOld, IVsHierarchy pHierNew, uint itemidNew, string pszMkDocumentNew)
{
return VSConstants.S_OK;
}
public int OnAfterDocumentWindowHide(uint docCookie, IVsWindowFrame pFrame)
{
return VSConstants.S_OK;
}
public int OnAfterFirstDocumentLock(uint docCookie, uint dwRDTLockType, uint dwReadLocksRemaining, uint dwEditLocksRemaining)
{
return VSConstants.S_OK;
}
public int OnAfterSave(uint docCookie)
{
return VSConstants.S_OK;
}
public int OnBeforeDocumentWindowShow(uint docCookie, int fFirstShow, IVsWindowFrame pFrame)
{
return VSConstants.S_OK;
}
public int OnBeforeLastDocumentUnlock(uint docCookie, uint dwRDTLockType, uint dwReadLocksRemaining, uint dwEditLocksRemaining)
{
return VSConstants.S_OK;
}
public int OnBeforeSave(uint docCookie)
{
if (null == BeforeSave)
return VSConstants.S_OK;
var document = FindDocumentByCookie(docCookie);
if (null == document)
return VSConstants.S_OK;
BeforeSave(this, FindDocumentByCookie(docCookie));
return VSConstants.S_OK;
}
#endregion
#region Private Methods
private Document FindDocumentByCookie(uint docCookie)
{
var documentInfo = mRunningDocumentTable.GetDocumentInfo(docCookie);
return mDte.Documents.Cast<Document>().FirstOrDefault(doc => doc.FullName == documentInfo.Moniker);
}
#endregion
}
当触发来自 VS 的任何类型的保存命令(CTRL + S、全部保存、编译、构建等)时,我使用此实现来格式化一些文档。
如果您想从某个命令(例如 Compile 获取保存事件,您必须在 OnBeforeSave() 方法中添加一个检查添加更多代码
首先,您必须保持对 CommandEvents var vommandEvents = dte.Events.CommandEvents
的强引用,并向 CommandEvents 添加一个新方法 commandEvents.BeforeExecute += CommandEventsBeforeExecute;
.
这会起作用,因为 CommandsEvents 将始终在 BeforeSave 之前调用。这就是 Visual Studio 中的工作原理,每个操作都代表一个需要一些步骤和事件的命令(例如,编译命令在其工作流程中包含保存文档事件)。
public override void OnBeforeSave(object sender, Document aDocument)
{
if (false == myCompileCommandFlag)
return;
// write your code here
}
public void CommandEventsBeforeExecute(string aGuid, int aId, object aCustomIn, object aCustomOut, ref bool aCancelDefault)
{
string commandName = GetCommandName(aGuid, aId);
if (0 != string.Compare("Build.Compile", commandName))
{
return;
}
myCompileCommandFlag= true;
}
public string GetCommandName(string aGuid, int aId)
{
if (null == aGuid)
return string.Empty;
if (null == mCommand)
return string.Empty;
Command cmd = mCommand.Item(aGuid, aId);
if (null == cmd)
return string.Empty;
return cmd.Name;
}