6

我的应用程序是一个 vb6 可执行文件,但系统中一些较新的形式是用 C# 编写的。我希望能够使用主应用程序窗口的句柄设置 C# 表单的 Owner 属性,以便在我的应用程序和其他应用程序之间来回切换时对话框保持在顶部。

我可以得到主应用程序窗口的 hwnd。我不确定我能从那里做什么?


2008 年 10 月 20 日 17:06 更新:

斯科特,

感谢您的回复。我忽略了 Show/ShowDialog 方法参数不是 Form 类型 - 我只查看 Owner 属性。

我从上面稍微修改了我正在使用的代码 - 我们有一个组件通常加载我们的表单并调用 ShowDialog。我的代码如下所示:

Form launchTarget = FormFactory.GetForm(xxx);  // psuedo-code for generic form loader
launchTarget.StartPosition = FormStartPosition.CenterParent;
IWin32Window parentWindow = GetWindowFromHwnd(hwnd);

launchTarget.ShowDialog(parentWindow);

GetWindowFromHwnd是您的代码的方法包装版本:

private IWin32Window GetWindowFromHost(int hwnd)
{
    IWin32Window window = null;
    IntPtr handle = new IntPtr(hwnd);

    try
    {
        NativeWindow nativeWindow = new NativeWindow();
        nativeWindow.AssignHandle(handle);
        window = nativeWindow;
    }
    finally
    {
        handle = IntPtr.Zero;
    }

    return window;
}

不幸的是,这并没有达到我的预期。该表单确实以模态方式显示,但是当我离开并返回父窗口时,它没有显示在正确的位置,也没有显示在顶部。我们的模态不会在任务栏中显示任务,因此窗口看似“消失”(尽管它仍然存在于 alt-tab 窗口列表中)。这对我来说表明我可能没有正确的 hwnd。如果您有任何其他建议,请回复。再次感谢。


更新 2008 年 11 月 10 日 16:25

一个后续评论 - 如果您将其分解为 try/finally 中的方法调用,如 Scott 的第二篇文章中那样,finally 块中的调用应该是:

parentWindow.ReleaseHandle();
4

2 回答 2

9

因此,您正在从 VB6 调用 C# Windows Form 类,这意味着您可能正在使用Show()or ShowDialog(),对吗?这两种方法都采用 IWin32Window 参数,该参数简单地定义了一个返回名为 Handle 的 IntPtr 属性的对象。

所以...您需要为您的 Windows 窗体类添加一个重载的构造函数(或 ShowDialog 方法),该类将 along作为参数,以便您可以将 VB6 hwnd 传递给窗体。一旦进入 C# 代码,您需要从 hwnd 创建一个 IntPtr 并将其分配给一个NativeWindow对象,然后将其作为所有者传递。

像这样的东西应该可以工作,尽管它未经测试:

public DialogResult ShowDialog(long hwnd)
{
   IntPtr handle = new IntPtr(hwnd);
   try
   {
      NativeWindow nativeWindow = new NativeWindow();

      nativeWindow.AssignHandle(handle);
      return this.ShowDialog(nativeWindow);
   }
   finally
   {
      handle = IntPtr.Zero;
   }
}
于 2008-10-17T20:36:13.303 回答
2

这太长了,不能作为评论发布......

我认为您遇到的问题是您包装我在 ShowDialog 重载中提供的代码的方式。如果您遵循您的GetWindowFromHost代码正在执行的操作,则会执行以下步骤:

  1. 从给定的 hwnd 创建一个新的 IntPtr。
  2. 创建一个新的 NativeWindow 对象并将其句柄分配为 IntPtr。
  3. 将 IntPtr(在 finally 块中)设置为 IntPtr.Zero。

我认为正是这个 finally 块给你带来了问题。在我的代码中,finally 块将在调用this.ShowDialog(nativeWindow)完成后运行。此时不再使用句柄 (IntPtr)。在您的代码中,您将返回一个IWin32Window仍应持有对该 IntPtr 的引用,在您调用该 IntPtr 时它launchTarget.ShowDialog(parentWindow)是 IntPtr.Zero。

尝试将您的代码更改为如下所示:

private NativeWindow GetWindowFromHost(int hwnd)
{
   IntPtr handle = new IntPtr(hwnd);
   NativeWindow nativeWindow = new NativeWindow();
   nativeWindow.AssignHandle(handle);
   return window;
}

然后将您的调用代码更改为如下所示:

Form launchTarget = FormFactory.GetForm(xxx);  // psuedo-code for generic form 
loaderlaunchTarget.StartPosition = FormStartPosition.CenterParent;
NativeWindow parentWindow = GetWindowFromHwnd(hwnd);

try
{
   launchTarget.ShowDialog(parentWindow);
}
finally
{
   parentWindow.DestroyHandle();
}

这些更改应该有效,但同样未经测试。

于 2008-10-20T17:27:56.617 回答