sdd*_*dds 10 vsto add-in ms-word office-interop customtaskpane
我正在使用Visual Studio 2008中的VSTO开发Word 2007-2010插件.在我的插件中,我需要为每个打开的word文档创建一个自定义任务窗格.基本上,我需要为每个文档创建一个任务窗格,在文档窗口中显示正确的任务窗格,在文档关闭时执行某些操作,然后删除任务窗格及其所有引用.
这是我到目前为止所做的:
任务窗格创建
我为每个新的,打开的或现有的加载文档创建一个自定义任务窗格,如下所示:
((ApplicationEvents4_Event) Application).NewDocument += CreateTaskPaneWrapper;
Application.DocumentOpen += CreateTaskPaneWrapper;
foreach (Document document in Application.Documents)
{
CreateTaskPaneWrapper(document);
}
Run Code Online (Sandbox Code Playgroud)
在CreateTaskPaneWrapper方法中,如果文档的任务窗格已存在,则检查Dictionary <Document,TaskPaneWrapper>.我这样做是因为如果我尝试打开已打开的文档,则会触发open事件.如果它不存在,我创建一个新的TaskPaneWrapper类.在其构造函数中,我创建了一个新的任务窗格,并将其添加到CustomTaskPanes集合中
Globals.ThisAddIn.CustomTaskPanes.Add(taskPane, "Title");
Run Code Online (Sandbox Code Playgroud)
根据MSDN,这将任务窗格与当前活动窗口相关联.
任务窗格关闭
Document.Close和Application.DocumentBeforeClose事件都不适合我,因为它们在用户发出关闭文档的确认之前触发.所以我在TaskPaneWrapper类中使用Microsoft.Office.Tools.Word.Document.Shutdown事件,如下所示:
_vstoDocument = document.GetVstoObject();
_vstoDocument.Shutdown += OnShutdown;
private void OnShutdown(object sender, EventArgs eventArgs)
{
Globals.ThisAddIn.CustomTaskPanes.Remove(_taskPane);
//additional shutdown logic
}
Run Code Online (Sandbox Code Playgroud)
所有这些似乎都很好用,创建了任务窗格,绑定到相应的窗口,并成功删除.但是,我仍有一个问题 - 当我启动Word时,会打开一个空白文档.如果我然后打开现有文档而不更改空白文档,则删除空白文档及其窗口,而不会触发Document.Close,Application.DocumentBeforeClose和Microsoft.Office.Tools.Word.Document.Shutdown事件.因为未调用OnShutdown并且未删除空白文档的任务窗格,所以下一个文档窗口包含两个任务窗格 - 非常新的任务窗格,以及第一个(孤立的)任务窗格.如何删除此孤立的任务窗格?访问已删除的文档或窗口引用会抛出COMException("对象已被删除").我暂时使用这个黑客:
//A property in my TaskPaneWrapper class
public bool IsWindowAlive()
{
try
{
var window = _vstoDocument.ActiveWindow;
return true;
}
catch (COMException)
{
return false;
}
}
Run Code Online (Sandbox Code Playgroud)
在CreateTaskPaneWrapper方法中,我检查此属性是否包含所有现有包装器,并关闭属性为false的那些包装器.当然,捕捉异常有点贵,而且这个解决方案非常糟糕,所以我想知道,有更好的解决方案吗?在这个问题中, CustomTaskPane.Window属性被检查为null,但它永远不会为我返回null.
另外,我可以使用我当前的逻辑遇到任何其他问题吗?管理多个文档的多个任务窗格的典型方法是什么?
小智 6
此MSDN文章" 在多个Word和InfoPath文档中管理任务窗格"中详细说明了此问题
您必须创建一个删除孤立CTP的方法(即那些不再附加窗口的CTP).
我已经按照这篇文章成功实现了一个删除孤儿的CustomTaskPane管理器.
小智 6
MSDN中的这个答案
您可以在调用 CreateTaskPaneWrapper 之前通过从 DocumentOpen 事件处理程序调用以下方法来主动清理,而不是在打开现有文档后被动清理孤立的任务窗格。此代码循环遍历属于加载项的每个自定义任务窗格。如果任务窗格没有关联的窗口,代码将从集合中删除任务窗格。
private void RemoveOrphanedTaskPanes()
{
for (int i = Globals.ThisAddIn.CustomTaskPanes.Count; i > 0; i--)
{
CustomTaskPanes ctp = Globals.ThisAddIn.CustomTaskPanes[i - 1];
if (ctp.Window == null)
{
this.CustomTaskPanes.Remove(ctp);
}
}
}
private void Application_DocumentOpen(Word.Document Doc)
{
RemoveOrphanedTaskPanes();
CreateTaskPaneWrapper(document);
}
Run Code Online (Sandbox Code Playgroud)
一些值得尝试的事情::
也许您可以检查 DocumentNew- 和 DocumentOpen-Event 中的“Word.Application.Documents.Count”来跟踪实际打开的文档。
要查明文档是否已打开(可能与重新打开活动文档相关),您可以使用打开文档的字典。键可以是唯一的文档名称。
您可以代替 IsWindowAlive() 方法来捕获 Dispose-Event,如下所示
_vstoDocument.Disposed += _vstoDocument_Disposed;
void _vstoDocument_Disposed(object sender, EventArgs e)
{
//Remove from dictionary of open documents
}
Run Code Online (Sandbox Code Playgroud)
归档时间: |
|
查看次数: |
6033 次 |
最近记录: |