当创建自定义任务窗格(_CustomTaskPane MSDN)并将其DockPosition设置为浮动时,我想指定出现的窗口的Top和Left属性。由于 Office COM API 不提供直接执行此操作的可能性,因此人们建议更改 CommandBar 相应属性的值:
var application = (_Application)_nativeTaskPane.Application;
application.CommandBars["Task Pane Title"].Top = top;
application.CommandBars["Task Pane Title"].Left = left;
在上面的代码中我假设
1) _nativeTaskPane 是我实现 _CustomTaskPane 的实例(实际上是 Microsoft.Office.Core.CustomTaskPane)
2) _应用程序是 Microsoft.Office.Interop.Excel._Application
当然,我是在设置Visible = true后这样做的。为了更加确定,甚至还订阅了任务窗格的 VisibleStateChange。但是我收到了带有 HRESULT E_FAILED 的 COMException。
问题是我可以在调试时读取这些属性(顶部和左侧),但是设置它们会引发异常。
看起来这个问题在互联网上至少出现过几次:
1) http://www.add-in-express.com/forum/read.php?FID=1&TID=5595
2)[http://aritrasaha.wordpress.com/2009/05/19/programatically-position-office-2007-floating-custom-task-pane/]
3)[http://www.visualstudiodev.com/visual-studio-tools-for-office/need-location-of-custom-task-pane-45822.shtml]
解决方法是使用 Windows API。但是,任何人都可以解释使用 CommandBar 方法可能会出现什么问题吗?也许我可以“重新配置”让这个上/左二传手毫无例外地工作。
这是我在程序中使用的解决方案:
/// <summary>
/// Set a custom panes position in the undocked state.
/// </summary>
/// <param name="customTaskPane">The custom task pane.</param>
/// <param name="x">The new X position.</param>
/// <param name="y">The new Y position.</param>
private void SetCustomPanePositionWhenFloating(CustomTaskPane customTaskPane, int x, int y)
{
var oldDockPosition = customTaskPane.DockPosition;
var oldVisibleState = customTaskPane.Visible;
customTaskPane.DockPosition = Microsoft.Office.Core.MsoCTPDockPosition.msoCTPDockPositionFloating;
customTaskPane.Visible = true; //The task pane must be visible to set its position
var window = FindWindowW("MsoCommandBar", customTaskPane.Title); //MLHIDE
if (window == null) return;
MoveWindow(window, x, y, customTaskPane.Width, customTaskPane.Height, true);
customTaskPane.Visible = oldVisibleState;
customTaskPane.DockPosition = oldDockPosition;
}
[DllImport("user32.dll", EntryPoint = "FindWindowW")]
public static extern System.IntPtr FindWindowW([System.Runtime.InteropServices.InAttribute()] [System.Runtime.InteropServices.MarshalAsAttribute(System.Runtime.InteropServices.UnmanagedType.LPWStr)] string lpClassName, [System.Runtime.InteropServices.InAttribute()] [System.Runtime.InteropServices.MarshalAsAttribute(System.Runtime.InteropServices.UnmanagedType.LPWStr)] string lpWindowName);
[DllImport("user32.dll", EntryPoint = "MoveWindow")]
[return: System.Runtime.InteropServices.MarshalAsAttribute(System.Runtime.InteropServices.UnmanagedType.Bool)]
public static extern bool MoveWindow([System.Runtime.InteropServices.InAttribute()] System.IntPtr hWnd, int X, int Y, int nWidth, int nHeight, [System.Runtime.InteropServices.MarshalAsAttribute(System.Runtime.InteropServices.UnmanagedType.Bool)] bool bRepaint);
/// <summary>
/// Set a custom panes size in the undocked state.
/// </summary>
/// <param name="customTaskPane">The custom task pane.</param>
/// <param name="width">The new width.</param>
/// <param name="height">The new height.</param>
private void SetCustomPaneSizeWhenFloating(CustomTaskPane customTaskPane, int width, int height)
{
var oldDockPosition = customTaskPane.DockPosition;
customTaskPane.DockPosition = Microsoft.Office.Core.MsoCTPDockPosition.msoCTPDockPositionFloating;
customTaskPane.Width = width;
customTaskPane.Height = height;
customTaskPane.DockPosition = oldDockPosition;
}
随意使用它...:-)
问候, 约尔格
它应该可以工作,并且 MVP Cindy Meister 对此错误有一个有趣的评论(她在回答这个论坛问题之前测试了它是否有效)- http://social.msdn.microsoft.com/Forums/vstudio/en-US/2df0e430-4d93-416e-89a0-56f8ad5dc988/seting-position-of-a-floating-custome-task-pane?prof=required
在其中,她说使用错误的变量来获取应用程序对象会导致错误,即:
Globals.MyAddIn.Application -> this will ultimately cause an exception
Globals.ThisAddin.Application -> this will ultimately work
我们假设两者返回相同的 Application 对象。
如果您认为这很奇怪,那么您就很优秀。
我在问题中添加了一条注释,询问为什么用于访问 Application 对象的变量名称会产生任何差异 - 当然它是使用相同的 Application 对象。
我怀疑这是由其他一些内部机制强加的一些可怕的内部、类似反射的限制。但无辜的开发者并没有受到这种非常奇怪的情况的保护。
我认为一旦自定义窗格设置为
floating
,您就无法根据定义更改其顶部/左侧属性。你到底想达到什么目的?您想将窗格放置在特定位置吗?如果是,请在将 visible
属性设置为 true 之前执行此操作
您应该引用的命令栏是“任务窗格”,它是 CommandBars 集合中的标准 CommandBar。您收到 HRESULT 消息是因为在 CommandBars 集合中找不到“任务窗格标题”CommandBar。