注意:虽然我的问题与VSTO Word post save event在同一个球场,但目标和目标(以及由此产生的所需代码)是不同的。 VSTO Word post save event的OP指出:
在将文档保存到磁盘后,我需要捕获该事件,关闭文件,执行我需要执行的操作并重新打开它。
我的需求是不同的。看我的OP。
注意结束
我有一个用于Word的VSTO
加载项,旨在操作RTF文件的各种元素(并且只有RTF文件)。加载项由功能区按钮调用。如果用户打开RTF文档,然后执行save-as
,我想捕获一个事件,以便我可以检查为save-as选择的文件名,并禁用调用我的加载项的按钮(如果扩展名不是.RTF
)。
在我的功能区类功能区加载方法(在我的功能区类的设计器文件中声明的事件处理方法:this.Load += new Microsoft.Office.Tools.Ribbon.RibbonUIEventHandler(this.Ribbon1_Load)
)中,我编写了各种可用事件(例如,Globals.ThisAddIn.Application.DocumentChange += Application_DocumentChange;
和Globals.ThisAddIn.Application.DocumentOpen += Application_DocumentOpen;
),但所有可用事件都在save-as
发生之前触发,不是之后。我还在这个带状加载方法中加了一个断点。保存之后不再执行(我并不感到惊讶)
我错过了什么吗?对于我的VSTO Word加载项,是否在save-as
事件之后触发了一个事件,该事件可以在我的功能区类中捕获,该事件将提供为save-as
选择的文件的名称?
更新我的代码反映了Cindy Meister的回答
感谢Microsoft Developer's Network上的Joseph Fox。我的代码来自Document Save Event
注意:我的VSTO功能区类名为ClsLesCaveat
。这是一个新组,有两个按钮,位于现有的Insert
表中。它仅使用VS Pro 2017中的VSTO设计器创建。
对我来说,我的功能区按钮需要在两种情况下禁用:
1)如果有人使用没有.RTF扩展名的Word打开文件,则应禁用我的功能区按钮
2)如果有人使用Word打开.RTF文件(我的按钮已启用),但如果他们对非.RTF文件执行保存,则应禁用该非-RTF文档的功能区按钮
注意:不关心保存,因为我的功能区按钮在打开时启用/禁用或另存为
using System;
using System.IO;
namespace LesCaveatAddIn
{
public partial class ThisAddIn
{
private bool allowSave = false;
private void ThisAddIn_Startup(object sender, System.EventArgs e)
{
this.Application.DocumentBeforeSave += Application_DocumentBeforeSave;
this.Application.DocumentOpen += Application_DocumentOpen;
}
# On open, disable buttons, enable buttons only if file extension is .RTF
private void Application_DocumentOpen(Microsoft.Office.Interop.Word.Document Doc)
{
string extension = (Path.GetExtension(Doc.FullName)).ToUpper();
Type type = typeof(ClsLesCaveat);
ClsLesCaveat ribbon = Globals.Ribbons.GetRibbon(type) as ClsLesCaveat;
ribbon.objButtonAddFouoCaveat.Enabled = false;
ribbon.objButtonAddLesCaveat.Enabled = false;
if (extension.Equals(".RTF"))
{
ribbon.objButtonAddFouoCaveat.Enabled = true;
ribbon.objButtonAddLesCaveat.Enabled = true;
}
}
# On save-as, handle the save-as myself. Cancel the save-as (since I just handled it). Then, disable buttons, enable buttons only if the save-as file extension is .RTF.
private void Application_DocumentBeforeSave(Microsoft.Office.Interop.Word.Document Doc, ref bool SaveAsUI, ref bool Cancel)
{
if (!allowSave)
{
allowSave = true;
if (SaveAsUI)
{
// Display Save As dialog
Microsoft.Office.Interop.Word.Dialog d = Globals.ThisAddIn.Application.Dialogs[Microsoft.Office.Interop.Word.WdWordDialog.wdDialogFileSaveAs];
object timeOut = 0;
d.Show(ref timeOut);
}
else
{
// Save without dialog
Doc.Save();
}
allowSave = false;
Cancel = true;
string extension = (Path.GetExtension(Doc.FullName)).ToUpper();
Type type = typeof(ClsLesCaveat);
ClsLesCaveat ribbon = Globals.Ribbons.GetRibbon(type) as ClsLesCaveat;
ribbon.objButtonAddFouoCaveat.Enabled = false;
ribbon.objButtonAddLesCaveat.Enabled = false;
if (extension.Equals(".RTF"))
{
ribbon.objButtonAddFouoCaveat.Enabled = true;
ribbon.objButtonAddLesCaveat.Enabled = true;
}
}
}
private void ThisAddIn_Shutdown(object sender, System.EventArgs e)
{
}
#region VSTO generated code
/// <summary>
/// Required method for Designer support - do not modify
/// the contents of this method with the code editor.
/// </summary>
private void InternalStartup()
{
this.Startup += new System.EventHandler(ThisAddIn_Startup);
this.Shutdown += new System.EventHandler(ThisAddIn_Shutdown);
}
#endregion
}
}
不,没有事件可以捕获任何保存或保存后的操作。与储蓄有关的唯一一个是DocumentBeforeSave。
DocumentBeforeSave确实提供了允许开发人员禁止内置UI(SaveAs对话框)以及取消触发事件的操作的参数。这允许开发人员提供他们自己的保存接口(as),这将确定文档何时被保存(as)并采取任何所需的操作,具体取决于文件名,扩展名或任何条件。
也可以使用Word的内置SaveAs对话框,而不是创建自己的对象框,尽管这在C#中有点四舍五入,因为它需要使用PInvoke。这是一个示例,让您了解这是如何工作的(未在移动设备上进行测试):
private void ThisDocument_BeforeSave(object sender, object e)
{
//Suppress the built-in SaveAs interface (dialog box)
e.SaveAsUi = false;
//Cancel the default action
e.Cancel = true;
Word.Dialog dlg = wdApplication.Dialogs[Microsoft.Office.Interop.Word.WdWordDialog.wdDialogFileSaveAs];
//Word dialog box parameters have to be accessed via Late-Binding (PInvoke)
//To get the path, use the Name property
object oDlg = (object)dlg;
object[] oArgs = new object[1];
oArgs[0] = (object)@"";
dlg.Show(ref missing);
object fileName = oDlg.GetType().InvokeMember("Name", BindingFlags.GetProperty, null, oDlg, oArgs);
}
here列出了可以使用的可用对话框参数。