Как использовать FolderBrowserDialog из приложения WPF

Я пытаюсь использовать FolderBrowserDialog из своего приложения WPF - ничего необычного. Меня не волнует, что Windows Forms обращаются к нему.

Однако, когда я вызываю ShowDialog, я хочу передать окно владельца, которое является IWin32Window. Как мне получить это из моего элемента управления WPF?

Собственно, это имеет значение? Если я запустил этот код и использую перегрузку ShowDialog без параметров, он отлично работает. При каких обстоятельствах мне нужно передать окно владельца?

Спасибо,

Craig

Ответ 1

И вот моя окончательная версия.

public static class MyWpfExtensions
{
    public static System.Windows.Forms.IWin32Window GetIWin32Window(this System.Windows.Media.Visual visual)
    {
        var source = System.Windows.PresentationSource.FromVisual(visual) as System.Windows.Interop.HwndSource;
        System.Windows.Forms.IWin32Window win = new OldWindow(source.Handle);
        return win;
    }

    private class OldWindow : System.Windows.Forms.IWin32Window
    {
        private readonly System.IntPtr _handle;
        public OldWindow(System.IntPtr handle)
        {
            _handle = handle;
        }

        #region IWin32Window Members
        System.IntPtr System.Windows.Forms.IWin32Window.Handle
        {
            get { return _handle; }
        }
        #endregion
    }
}

И на самом деле его использовать:

var dlg = new FolderBrowserDialog();
System.Windows.Forms.DialogResult result = dlg.ShowDialog(this.GetIWin32Window());

Ответ 2

Если вы укажете Owner, вы получите диалог Modal над указанным окном WPF.

Чтобы получить совместимое с WinForms окно Win32, создайте класс, реализующий IWin32Window, как этот

 public class OldWindow : System.Windows.Forms.IWin32Window
{
    IntPtr _handle;

    public OldWindow(IntPtr handle)
    {
        _handle = handle;
    }

    #region IWin32Window Members

    IntPtr System.Windows.Forms.IWin32Window.Handle
    {
        get { return _handle; }
    }

    #endregion
}

И используйте экземпляр этого класса в WinForms

        IntPtr mainWindowPtr = new WindowInteropHelper(this).Handle; // 'this' means WPF Window
        folderBrowserDialog.ShowDialog(new OldWindow(mainWindowPtr));

Ответ 3

Я понимаю, что это старый вопрос, но вот подход, который может быть немного более изящным (и может быть или не быть доступен раньше)...

using System;
using System.Windows;
using System.Windows.Forms;

// ...

/// <summary>
///     Utilities for easier integration with WinForms.
/// </summary>
public static class WinFormsCompatibility {

    /// <summary>
    ///     Gets a handle of the given <paramref name="window"/> and wraps it into <see cref="IWin32Window"/>,
    ///     so it can be consumed by WinForms code, such as <see cref="FolderBrowserDialog"/>.
    /// </summary>
    /// <param name="window">
    ///     The WPF window whose handle to get.
    /// </param>
    /// <returns>
    ///     The handle of <paramref name="window"/> is returned as <see cref="IWin32Window.Handle"/>.
    /// </returns>
    public static IWin32Window GetIWin32Window(this Window window) {
        return new Win32Window(new System.Windows.Interop.WindowInteropHelper(window).Handle);
    }

    /// <summary>
    ///     Implementation detail of <see cref="GetIWin32Window"/>.
    /// </summary>
    class Win32Window : IWin32Window { // NOTE: This is System.Windows.Forms.IWin32Window, not System.Windows.Interop.IWin32Window!

        public Win32Window(IntPtr handle) {
            Handle = handle; // C# 6 "read-only" automatic property.
        }

        public IntPtr Handle { get; }

    }

}

Затем из окна WPF вы можете просто...

public partial class MainWindow : Window {

    void Button_Click(object sender, RoutedEventArgs e) {
        using (var dialog = new FolderBrowserDialog()) {
            if (dialog.ShowDialog(this.GetIWin32Window()) == System.Windows.Forms.DialogResult.OK) {
                // Use dialog.SelectedPath.
            }
        }
    }

}

Собственно, это имеет значение?

Я не уверен, имеет ли это значение в этом случае, но в целом вы должны указать Windows, какова ваша иерархия окон, поэтому, если щелкнуть родительское окно, когда дочернее окно является модальным, Windows может предоставить визуальный (и, возможно, слышимый ) подскажите пользователю.

Кроме того, он гарантирует, что "правильное" окно находится сверху, когда есть несколько модальных окон (а не то, что я сторонник такого дизайна пользовательского интерфейса). Я видел пользовательские интерфейсы, разработанные определенной многомиллиардной корпорацией (оболочка которой остается неназванной), которая висела просто потому, что один модальный диалог "застрял" под другим, и пользователь понятия не имел, что это было даже там, не говоря уже о том, как закрыть он.

Ответ 4

ОК, понял это сейчас - благодаря Джоби, чей ответ был близок, но не совсем.

Из приложения WPF, здесь мой код, который работает:

Сначала вспомогательный класс:

private class OldWindow : System.Windows.Forms.IWin32Window
{    
    IntPtr _handle;    
    public OldWindow(IntPtr handle)
    {
        _handle = handle;
    }   

    #region IWin32Window Members    
    IntPtr System.Windows.Forms.IWin32Window.Handle
    {
        get { return _handle; }
    }    
    #endregion
}

Затем, чтобы использовать это:

    System.Windows.Forms.FolderBrowserDialog dlg = new FolderBrowserDialog();
    HwndSource source = PresentationSource.FromVisual(this) as HwndSource;
    System.Windows.Forms.IWin32Window win = new OldWindow(source.Handle);
    System.Windows.Forms.DialogResult result = dlg.ShowDialog(win);

Я уверен, что смогу обернуть это лучше, но в основном это работает. Ура!: -)

Ответ 5

//add a reference to System.Windows.Forms.dll

public partial class MainWindow : Window, System.Windows.Forms.IWin32Window
{
    public MainWindow()
    {
        InitializeComponent();
    }

    private void button_Click(object sender, RoutedEventArgs e)
    {
        var fbd = new FolderBrowserDialog();
        fbd.ShowDialog(this);
    }

    IntPtr System.Windows.Forms.IWin32Window.Handle
    {
        get
        {
            return ((HwndSource)PresentationSource.FromVisual(this)).Handle;
        }
    }
}

Ответ 6

Перевод VB.net

Module MyWpfExtensions

Public Function GetIWin32Window(this As Object, visual As System.Windows.Media.Visual) As System.Windows.Forms.IWin32Window

    Dim source As System.Windows.Interop.HwndSource = System.Windows.PresentationSource.FromVisual(Visual)
    Dim win As System.Windows.Forms.IWin32Window = New OldWindow(source.Handle)
    Return win
End Function

Private Class OldWindow
    Implements System.Windows.Forms.IWin32Window

    Public Sub New(handle As System.IntPtr)
        _handle = handle
    End Sub


    Dim _handle As System.IntPtr
    Public ReadOnly Property Handle As IntPtr Implements Forms.IWin32Window.Handle
        Get

        End Get
    End Property


End Class

End Module

Ответ 7

Преимущество передачи дескриптора владельца заключается в том, что FolderBrowserDialog не будет модальным для этого окна. Это предотвращает взаимодействие пользователя с вашим основным окном приложения при активном диалоге.

Ответ 8

Вы должны получить IWin32Window с помощью PresentationSource.FromVisual и выдать результат HwndSource, который реализует IWin32Window.

Также в комментариях здесь:

Ответ 9

Почему бы не использовать встроенный класс WindowInteropHelper (см. пространство имен System.Windows.Interop). Этот класс уже использует IWin32Window;)

Итак, вы можете забыть о "классе OldWindow"... использование остается тем же

Ответ 10

Вот простой метод.


System.Windows.Forms.NativeWindow winForm; 
public MainWindow()
{
    winForm = new System.Windows.Forms.NativeWindow();
    winForm.AssignHandle(new WindowInteropHelper(this).Handle);
    ...
}
public showDialog()
{
   dlgFolderBrowser.ShowDialog(winForm);
}