using Avalonia; using Avalonia.Controls; using Avalonia.Controls.ApplicationLifetimes; using Avalonia.Media; using Avalonia.Threading; using FluentAvalonia.Core; using FluentAvalonia.UI.Controls; using Ryujinx.Ava.Common.Locale; using Ryujinx.Ava.UI.Controls; using Ryujinx.Ava.UI.Windows; using Ryujinx.Common.Logging; using System; using System.Threading; using System.Threading.Tasks; namespace Ryujinx.Ava.UI.Helpers { public static class ContentDialogHelper { private static bool _isChoiceDialogOpen; public async static Task<UserResult> ShowContentDialog( string title, object content, string primaryButton, string secondaryButton, string closeButton, UserResult primaryButtonResult = UserResult.Ok, ManualResetEvent deferResetEvent = null, Func<Window, Task> doWhileDeferred = null, TypedEventHandler<ContentDialog, ContentDialogButtonClickEventArgs> deferCloseAction = null) { UserResult result = UserResult.None; ContentDialog contentDialog = new() { Title = title, PrimaryButtonText = primaryButton, SecondaryButtonText = secondaryButton, CloseButtonText = closeButton, Content = content }; contentDialog.PrimaryButtonCommand = MiniCommand.Create(() => { result = primaryButtonResult; }); contentDialog.SecondaryButtonCommand = MiniCommand.Create(() => { result = UserResult.No; contentDialog.PrimaryButtonClick -= deferCloseAction; }); contentDialog.CloseButtonCommand = MiniCommand.Create(() => { result = UserResult.Cancel; contentDialog.PrimaryButtonClick -= deferCloseAction; }); if (deferResetEvent != null) { contentDialog.PrimaryButtonClick += deferCloseAction; } await ShowAsync(contentDialog); return result; } private async static Task<UserResult> ShowTextDialog( string title, string primaryText, string secondaryText, string primaryButton, string secondaryButton, string closeButton, int iconSymbol, UserResult primaryButtonResult = UserResult.Ok, ManualResetEvent deferResetEvent = null, Func<Window, Task> doWhileDeferred = null, TypedEventHandler<ContentDialog, ContentDialogButtonClickEventArgs> deferCloseAction = null) { Grid content = CreateTextDialogContent(primaryText, secondaryText, iconSymbol); return await ShowContentDialog(title, content, primaryButton, secondaryButton, closeButton, primaryButtonResult, deferResetEvent, doWhileDeferred, deferCloseAction); } public async static Task<UserResult> ShowDeferredContentDialog( StyleableWindow window, string title, string primaryText, string secondaryText, string primaryButton, string secondaryButton, string closeButton, int iconSymbol, ManualResetEvent deferResetEvent, Func<Window, Task> doWhileDeferred = null) { bool startedDeferring = false; UserResult result = UserResult.None; return await ShowTextDialog( title, primaryText, secondaryText, primaryButton, secondaryButton, closeButton, iconSymbol, primaryButton == LocaleManager.Instance[LocaleKeys.InputDialogYes] ? UserResult.Yes : UserResult.Ok, deferResetEvent, doWhileDeferred, DeferClose); async void DeferClose(ContentDialog sender, ContentDialogButtonClickEventArgs args) { if (startedDeferring) { return; } sender.PrimaryButtonClick -= DeferClose; startedDeferring = true; var deferral = args.GetDeferral(); result = primaryButton == LocaleManager.Instance[LocaleKeys.InputDialogYes] ? UserResult.Yes : UserResult.Ok; sender.PrimaryButtonClick -= DeferClose; _ = Task.Run(() => { deferResetEvent.WaitOne(); Dispatcher.UIThread.Post(() => { deferral.Complete(); }); }); if (doWhileDeferred != null) { await doWhileDeferred(window); deferResetEvent.Set(); } } } private static Grid CreateTextDialogContent(string primaryText, string secondaryText, int symbol) { Grid content = new() { RowDefinitions = new RowDefinitions() { new RowDefinition(), new RowDefinition() }, ColumnDefinitions = new ColumnDefinitions() { new ColumnDefinition(GridLength.Auto), new ColumnDefinition() }, MinHeight = 80 }; SymbolIcon icon = new() { Symbol = (Symbol)symbol, Margin = new Thickness(10), FontSize = 40, VerticalAlignment = Avalonia.Layout.VerticalAlignment.Center }; Grid.SetColumn(icon, 0); Grid.SetRowSpan(icon, 2); Grid.SetRow(icon, 0); TextBlock primaryLabel = new() { Text = primaryText, Margin = new Thickness(5), TextWrapping = TextWrapping.Wrap, MaxWidth = 450 }; TextBlock secondaryLabel = new() { Text = secondaryText, Margin = new Thickness(5), TextWrapping = TextWrapping.Wrap, MaxWidth = 450 }; Grid.SetColumn(primaryLabel, 1); Grid.SetColumn(secondaryLabel, 1); Grid.SetRow(primaryLabel, 0); Grid.SetRow(secondaryLabel, 1); content.Children.Add(icon); content.Children.Add(primaryLabel); content.Children.Add(secondaryLabel); return content; } public static async Task<UserResult> CreateInfoDialog( string primary, string secondaryText, string acceptButton, string closeButton, string title) { return await ShowTextDialog( title, primary, secondaryText, acceptButton, "", closeButton, (int)Symbol.Important); } internal static async Task<UserResult> CreateConfirmationDialog( string primaryText, string secondaryText, string acceptButtonText, string cancelButtonText, string title, UserResult primaryButtonResult = UserResult.Yes) { return await ShowTextDialog( string.IsNullOrWhiteSpace(title) ? LocaleManager.Instance[LocaleKeys.DialogConfirmationTitle] : title, primaryText, secondaryText, acceptButtonText, "", cancelButtonText, (int)Symbol.Help, primaryButtonResult); } internal static UpdateWaitWindow CreateWaitingDialog(string mainText, string secondaryText) { return new(mainText, secondaryText); } internal static async Task CreateUpdaterInfoDialog(string primary, string secondaryText) { await ShowTextDialog( LocaleManager.Instance[LocaleKeys.DialogUpdaterTitle], primary, secondaryText, "", "", LocaleManager.Instance[LocaleKeys.InputDialogOk], (int)Symbol.Important); } internal static async Task CreateWarningDialog(string primary, string secondaryText) { await ShowTextDialog( LocaleManager.Instance[LocaleKeys.DialogWarningTitle], primary, secondaryText, "", "", LocaleManager.Instance[LocaleKeys.InputDialogOk], (int)Symbol.Important); } internal static async Task CreateErrorDialog(string errorMessage, string secondaryErrorMessage = "") { Logger.Error?.Print(LogClass.Application, errorMessage); await ShowTextDialog( LocaleManager.Instance[LocaleKeys.DialogErrorTitle], LocaleManager.Instance[LocaleKeys.DialogErrorMessage], errorMessage, secondaryErrorMessage, "", LocaleManager.Instance[LocaleKeys.InputDialogOk], (int)Symbol.Dismiss); } internal static async Task<bool> CreateChoiceDialog(string title, string primary, string secondaryText) { if (_isChoiceDialogOpen) { return false; } _isChoiceDialogOpen = true; UserResult response = await ShowTextDialog( title, primary, secondaryText, LocaleManager.Instance[LocaleKeys.InputDialogYes], "", LocaleManager.Instance[LocaleKeys.InputDialogNo], (int)Symbol.Help, UserResult.Yes); _isChoiceDialogOpen = false; return response == UserResult.Yes; } internal static async Task<bool> CreateExitDialog() { return await CreateChoiceDialog( LocaleManager.Instance[LocaleKeys.DialogExitTitle], LocaleManager.Instance[LocaleKeys.DialogExitMessage], LocaleManager.Instance[LocaleKeys.DialogExitSubMessage]); } internal static async Task<bool> CreateStopEmulationDialog() { return await CreateChoiceDialog( LocaleManager.Instance[LocaleKeys.DialogStopEmulationTitle], LocaleManager.Instance[LocaleKeys.DialogStopEmulationMessage], LocaleManager.Instance[LocaleKeys.DialogExitSubMessage]); } internal static async Task<string> CreateInputDialog( string title, string mainText, string subText, uint maxLength = int.MaxValue, string input = "") { var result = await InputDialog.ShowInputDialog( title, mainText, input, subText, maxLength); if (result.Result == UserResult.Ok) { return result.Input; } return string.Empty; } public static async Task<ContentDialogResult> ShowAsync(ContentDialog contentDialog) { ContentDialogResult result; ContentDialogOverlayWindow contentDialogOverlayWindow = null; Window parent = GetMainWindow(); if (parent.IsActive && parent is MainWindow window && window.ViewModel.IsGameRunning) { contentDialogOverlayWindow = new() { Height = parent.Bounds.Height, Width = parent.Bounds.Width, Position = parent.PointToScreen(new Point()), ShowInTaskbar = false }; parent.PositionChanged += OverlayOnPositionChanged; void OverlayOnPositionChanged(object sender, PixelPointEventArgs e) { contentDialogOverlayWindow.Position = parent.PointToScreen(new Point()); } contentDialogOverlayWindow.ContentDialog = contentDialog; bool opened = false; contentDialogOverlayWindow.Opened += OverlayOnActivated; async void OverlayOnActivated(object sender, EventArgs e) { if (opened) { return; } opened = true; contentDialogOverlayWindow.Position = parent.PointToScreen(new Point()); result = await ShowDialog(); } result = await contentDialogOverlayWindow.ShowDialog<ContentDialogResult>(parent); } else { result = await ShowDialog(); } async Task<ContentDialogResult> ShowDialog() { if (contentDialogOverlayWindow is not null) { result = await contentDialog.ShowAsync(contentDialogOverlayWindow); contentDialogOverlayWindow!.Close(); } else { result = await contentDialog.ShowAsync(); } return result; } if (contentDialogOverlayWindow is not null) { contentDialogOverlayWindow.Content = null; contentDialogOverlayWindow.Close(); } return result; } private static Window GetMainWindow() { if (Application.Current.ApplicationLifetime is IClassicDesktopStyleApplicationLifetime al) { foreach (Window item in al.Windows) { if (item.IsActive && item is MainWindow window) { return window; } } } return null; } } }