在 UI 线程上调用异步方法

Call async method on UI thread

我正在尝试创建具有 IdentityServer 身份验证的 WPF 客户端。我正在使用他们的 OidcClient 登录。它是完全异步的,而我的应用程序是同步的,如果不付出巨大的努力就无法重构。打电话

var result = await _oidcClient.LoginAsync();

不等待结果。调用 Wait().Result 会导致死锁。将它包装到其他 Task.Run 抱怨该方法不是 UI 线程上的 运行(它打开带有登录对话框的浏览器)。

你有什么想法,如何解决这个问题?我需要编写自定义同步 OidcClient 吗?

对于需要在不进行大量重构的情况下向遗留应用程序引入异步的其他类似情况,我建议使用简单的 "Please wait..." 模态对话框。该对话框启动异步操作并在操作完成后自行关闭。

Window.ShowDialog 是一个同步的 API 方式,它阻塞主 UI 并且只有 returns 给调用者,当模态对话框关闭时。但是,它仍然运行一个嵌套的消息循环并发送消息。因此,异步任务继续回调仍然会被抽取和执行,而不是使用容易死锁的 Task.Wait().

这是一个基本但完整的 WPF 示例,用 Task.Delay() 模拟 _oidcClient.LoginAsync() 并在 UI 线程上执行它,有关详细信息,请参阅 WpfTaskExt.Execute

取消支持是可选的;如果无法取消实际 LoginAsync,则防止对话框过早关闭。

using System;
using System.Threading;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;

namespace WpfApp1
{
    public partial class MainWindow : Window
    {
        public MainWindow()
        {
            var button = new Button() { Content = "Login", Width = 100, Height = 20 };
            button.Click += HandleLogin;
            this.Content = button;
        }

        // simulate _oidcClient.LoginAsync
        static async Task<bool> LoginAsync(CancellationToken token)
        {
            await Task.Delay(5000, token);
            return true;
        }

        void HandleLogin(object sender, RoutedEventArgs e)
        {
            try
            {
                var result = WpfTaskExt.Execute(
                    taskFunc: token => LoginAsync(token),
                    createDialog: () =>
                        new Window
                        {
                            Owner = this,
                            Width = 320,
                            Height = 200,
                            WindowStartupLocation = WindowStartupLocation.CenterOwner,
                            Content = new TextBox
                            {
                                Text = "Loggin in, please wait... ",
                                HorizontalContentAlignment = HorizontalAlignment.Center,
                                VerticalContentAlignment = VerticalAlignment.Center
                            },
                            WindowStyle = WindowStyle.ToolWindow
                        },
                    token: CancellationToken.None);

                MessageBox.Show($"Success: {result}");
            }
            catch (Exception ex)
            {
                MessageBox.Show(ex.Message);
            }
        }
    }

    public static class WpfTaskExt
    {
        /// <summary>
        /// Execute an async func synchronously on a UI thread,
        /// on a modal dialog's nested message loop
        /// </summary>
        public static TResult Execute<TResult>(
            Func<CancellationToken, Task<TResult>> taskFunc,
            Func<Window> createDialog,
            CancellationToken token = default(CancellationToken))
        {
            var cts = CancellationTokenSource.CreateLinkedTokenSource(token);

            var dialog = createDialog();
            var canClose = false;
            Task<TResult> task = null;

            async Task<TResult> taskRunner()
            {
                try
                {
                    return await taskFunc(cts.Token);
                }
                finally
                {
                    canClose = true;
                    if (dialog.IsLoaded)
                    {
                        dialog.Close();
                    }
                }
            }

            dialog.Closing += (_, args) =>
            {
                if (!canClose)
                {
                    args.Cancel = true; // must stay open for now
                    cts.Cancel();
                }
            };

            dialog.Loaded += (_, __) =>
            {
                task = taskRunner();
            };

            dialog.ShowDialog();

            return task.GetAwaiter().GetResult();
        }
    }
}