C#。 Excel 插件。无法重新定位浮动自定义任务窗格

问题描述 投票:0回答:4

当创建自定义任务窗格(_CustomTaskPane MSDN)并将其DockPosition设置为浮动时,我想指定出现的窗口的TopLeft属性。由于 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 方法可能会出现什么问题吗?也许我可以“重新配置”让这个上/左二传手毫无例外地工作。

c# excel add-in excel-addins customtaskpane
4个回答
4
投票

这是我在程序中使用的解决方案:

    /// <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;
    }

随意使用它...:-)

问候, 约尔格


1
投票

它应该可以工作,并且 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 对象。

我怀疑这是由其他一些内部机制强加的一些可怕的内部、类似反射的限制。但无辜的开发者并没有受到这种非常奇怪的情况的保护。


0
投票

我认为一旦自定义窗格设置为

floating
,您就无法根据定义更改其顶部/左侧属性。你到底想达到什么目的?您想将窗格放置在特定位置吗?如果是,请在将
visible
属性设置为 true

之前执行此操作

0
投票

您应该引用的命令栏是“任务窗格”,它是 CommandBars 集合中的标准 CommandBar。您收到 HRESULT 消息是因为在 CommandBars 集合中找不到“任务窗格标题”CommandBar。

© www.soinside.com 2019 - 2024. All rights reserved.