2

I have a simple WPF application MyWPF.exe,

namespace MyWPF
{
     /// <summary>
     /// Interaction logic for MainWindow.xaml
     /// </summary>
   public partial class MainWindow : Window
   {           

    public MainWindow()
    {
        InitializeComponent();

    }

    private void button1_Click(object sender, RoutedEventArgs e)
    {
     MessageBox.Show("Button is clicked");
    }
  }
}

I want to load this "MainWindow" in my Winform application, I tried like below, I am getting exception when creating instance as "System.InvalidOperationException: Window must be the root of the tree. Cannot add Window as a child of Visual."

        String path = @"D:\Test\MyForm\MyWPF\bin\Debug\MyWPF.exe";
        System.Reflection.Assembly myExeApp = System.Reflection.Assembly.LoadFile(path);
        System.Type MyWPFType = myExeApp.GetType("MyWPF.MainWindow");

        var MyWPFInstance = (System.Windows.Window)myExeApp.CreateInstance("MyWPF.MainWindow");

        ctrlHost = new ElementHost();
        ctrlHost.Dock = DockStyle.Fill;
        ElementHost.EnableModelessKeyboardInterop(MyWPFInstance );
        ctrlHost.Child = MyWPFInstance ;
        this.Controls.Add(ctrlHost);

Is it possible or am I missing something? The entire exception is as below,

System.InvalidOperationException: Window must be the root of the tree. Cannot add Window as a child of Visual.
   at System.Windows.Window.OnAncestorChanged()
   at System.Windows.FrameworkElement.OnAncestorChangedInternal(TreeChangeInfo parentTreeState)
   at System.Windows.TreeWalkHelper.OnAncestorChanged(DependencyObject d, TreeChangeInfo info)
   at System.Windows.DescendentsWalker`1.StartWalk(DependencyObject startNode, Boolean skipStartNode)
   at MS.Internal.PrePostDescendentsWalker`1.StartWalk(DependencyObject startNode, Boolean skipStartNode)
   at System.Windows.TreeWalkHelper.InvalidateOnTreeChange(FrameworkElement fe, FrameworkContentElement fce, DependencyObject parent, Boolean isAddOperation)
   at System.Windows.FrameworkElement.ChangeLogicalParent(DependencyObject newParent)
   at System.Windows.FrameworkElement.AddLogicalChild(Object child)
   at System.Windows.Controls.UIElementCollection.AddInternal(UIElement element)
   at System.Windows.Controls.UIElementCollection.Add(UIElement element)
   at System.Windows.Forms.Integration.ElementHost.set_Child(UIElement value)
   at MyForm.Form1.button1_Click(Object sender, EventArgs e) in D:\Test\MyForm\MyForm\Form1.cs:line 37
   at System.Windows.Forms.Control.OnClick(EventArgs e)
   at System.Windows.Forms.Button.OnClick(EventArgs e)
   at System.Windows.Forms.Button.OnMouseUp(MouseEventArgs mevent)
   at System.Windows.Forms.Control.WmMouseUp(Message& m, MouseButtons button, Int32 clicks)
   at System.Windows.Forms.Control.WndProc(Message& m)
   at System.Windows.Forms.ButtonBase.WndProc(Message& m)
   at System.Windows.Forms.Button.WndProc(Message& m)
   at System.Windows.Forms.Control.ControlNativeWindow.OnMessage(Message& m)
   at System.Windows.Forms.Control.ControlNativeWindow.WndProc(Message& m)
   at System.Windows.Forms.NativeWindow.Callback(IntPtr hWnd, Int32 msg, IntPtr wparam, IntPtr lparam)
4

2 回答 2

2

ElementHost不适用于这种互操作。您可以使用任何非Window控件作为其子项。您可以重构您的 WPF 应用程序,以便主窗口包含一个用户控件,您将使用反射代码加载该控件并将其添加到表单中。

或者,您可以使用 Win32SetParent将 WPF 窗口放置为 Windows 窗体控件的子级,但我绝对会避免这种情况。

于 2013-10-03T13:36:36.620 回答
0

我知道,这是一个老问题,但也许其他人也在寻找解决方案。

只需 在您的每个 WPF 窗口中输入: ElementHost.EnableModelessKeyboardInterop(this); before即可。InitializeComponent();

它帮助了我。

于 2020-08-04T06:56:19.893 回答