using System; using System.Linq; using CommunityToolkit.Mvvm.Messaging; using Microsoft.UI.Xaml; using Microsoft.UI.Xaml.Controls; using Wino.Calendar.Views; using Wino.Core.Domain.Enums; using Wino.Core.Domain.Interfaces; using Wino.Core.Domain.Models.Calendar; using Wino.Core.Domain.Models.Navigation; using Wino.Helpers; using Wino.Mail.ViewModels.Data; using Wino.Mail.ViewModels.Messages; using Wino.Mail.WinUI; using Wino.Mail.WinUI.Interfaces; using Wino.Mail.WinUI.Models; using Wino.Mail.WinUI.Services; using Wino.Mail.WinUI.Views.Calendar; using Wino.Messaging.Client.Mails; using Wino.Messaging.Client.Calendar; using Wino.Messaging.Client.Navigation; using Wino.Views; using Wino.Views.Account; using Wino.Views.Mail; using Wino.Views.Settings; namespace Wino.Services; public class NavigationService : NavigationServiceBase, INavigationService { private readonly IStatePersistanceService _statePersistanceService; private readonly IDispatcher _dispatcher; private readonly IWinoWindowManager _windowManager; private WinoPage[] _renderingPageTypes = new WinoPage[] { WinoPage.MailRenderingPage, WinoPage.ComposePage }; private static readonly WinoPage[] MailOnlyPages = [ WinoPage.MailListPage, WinoPage.MailRenderingPage, WinoPage.ComposePage, WinoPage.IdlePage, WinoPage.WelcomePage, WinoPage.WelcomePageV2, WinoPage.WelcomeHostPage, WinoPage.ProviderSelectionPage, WinoPage.AccountSetupProgressPage, WinoPage.SpecialImapCredentialsPage ]; private static readonly WinoPage[] CalendarOnlyPages = [ WinoPage.CalendarPage, WinoPage.EventDetailsPage, WinoPage.CalendarEventComposePage ]; public NavigationService(IStatePersistanceService statePersistanceService, IDispatcher dispatcher, IWinoWindowManager windowManager) { _statePersistanceService = statePersistanceService; _dispatcher = dispatcher; _windowManager = windowManager; } private bool IsOnNavigationThread() => _dispatcher is WinUIDispatcher winUiDispatcher && winUiDispatcher.HasThreadAccess; private T ExecuteOnNavigationThread(Func action) { if (IsOnNavigationThread()) return action(); T result = default!; _dispatcher.ExecuteOnUIThread(() => result = action()).GetAwaiter().GetResult(); return result; } private void ExecuteOnNavigationThread(Action action) { if (IsOnNavigationThread()) { action(); return; } _dispatcher.ExecuteOnUIThread(action).GetAwaiter().GetResult(); } public Type? GetPageType(WinoPage winoPage) { return winoPage switch { WinoPage.None => null, WinoPage.IdlePage => typeof(IdlePage), WinoPage.AccountDetailsPage => typeof(AccountDetailsPage), WinoPage.MergedAccountDetailsPage => typeof(MergedAccountDetailsPage), WinoPage.AccountManagementPage => typeof(AccountManagementPage), WinoPage.ManageAccountsPage => typeof(ManageAccountsPage), WinoPage.SignatureManagementPage => typeof(SignatureManagementPage), WinoPage.AboutPage => typeof(AboutPage), WinoPage.PersonalizationPage => typeof(PersonalizationPage), WinoPage.MessageListPage => typeof(MessageListPage), WinoPage.ReadComposePanePage => typeof(ReadComposePanePage), WinoPage.MailRenderingPage => typeof(MailRenderingPage), WinoPage.ComposePage => typeof(ComposePage), WinoPage.MailListPage => typeof(MailListPage), WinoPage.SettingsPage => typeof(SettingsPage), WinoPage.WelcomePage => typeof(WelcomePage), WinoPage.WelcomePageV2 => typeof(WelcomePageV2), WinoPage.SettingOptionsPage => typeof(SettingOptionsPage), WinoPage.AppPreferencesPage => typeof(AppPreferencesPage), WinoPage.AliasManagementPage => typeof(AliasManagementPage), WinoPage.LanguageTimePage => typeof(LanguageTimePage), WinoPage.EditAccountDetailsPage => typeof(EditAccountDetailsPage), WinoPage.ImapCalDavSettingsPage => typeof(ImapCalDavSettingsPage), WinoPage.KeyboardShortcutsPage => typeof(KeyboardShortcutsPage), WinoPage.ContactsPage => typeof(ContactsPage), WinoPage.SignatureAndEncryptionPage => typeof(SignatureAndEncryptionPage), WinoPage.EmailTemplatesPage => typeof(EmailTemplatesPage), WinoPage.CreateEmailTemplatePage => typeof(CreateEmailTemplatePage), WinoPage.StoragePage => typeof(StoragePage), WinoPage.WelcomeHostPage => typeof(WelcomeHostPage), WinoPage.ProviderSelectionPage => typeof(ProviderSelectionPage), WinoPage.AccountSetupProgressPage => typeof(AccountSetupProgressPage), WinoPage.SpecialImapCredentialsPage => typeof(SpecialImapCredentialsPage), WinoPage.CalendarPage => typeof(CalendarPage), WinoPage.EventDetailsPage => typeof(EventDetailsPage), WinoPage.CalendarEventComposePage => typeof(CalendarEventComposePage), WinoPage.CalendarSettingsPage => typeof(CalendarSettingsPage), WinoPage.CalendarAccountSettingsPage => typeof(CalendarAccountSettingsPage), _ => null, }; } public Frame GetCoreFrame(NavigationReferenceFrame frameType) => ExecuteOnNavigationThread(() => GetCoreFrameInternal(frameType) ?? throw new ArgumentException($"Frame '{frameType}' cannot be resolved.")); private Frame? GetCoreFrameInternal(NavigationReferenceFrame frameType, WinoWindowKind? requestedWindowKind = null) { if (frameType == NavigationReferenceFrame.ShellFrame) { if (requestedWindowKind.HasValue) return _windowManager.GetPrimaryNavigationFrame(requestedWindowKind.Value); var activeWindow = _windowManager.ActiveWindow; if (activeWindow != null) { var activeShellWindow = _windowManager.GetWindow(WinoWindowKind.Shell); if (ReferenceEquals(activeWindow, activeShellWindow)) return _windowManager.GetPrimaryNavigationFrame(WinoWindowKind.Shell); var activeWelcomeWindow = _windowManager.GetWindow(WinoWindowKind.Welcome); if (ReferenceEquals(activeWindow, activeWelcomeWindow)) return _windowManager.GetPrimaryNavigationFrame(WinoWindowKind.Welcome); } return _windowManager.GetPrimaryNavigationFrame(WinoWindowKind.Shell) ?? _windowManager.GetPrimaryNavigationFrame(WinoWindowKind.Welcome); } var mainFrame = _windowManager.GetPrimaryNavigationFrame(WinoWindowKind.Shell); if (mainFrame == null) return null; var contentRoot = mainFrame.Content as FrameworkElement; if (contentRoot == null) return null; // Use FindName first — it works immediately after InitializeComponent(), // before the visual tree is built by the layout pass. if (contentRoot.FindName(frameType.ToString()) is Frame namedFrame) return namedFrame; // Fall back to visual tree search for deeply nested frames (e.g. RenderingFrame). return WinoVisualTreeHelper.GetChildObject(contentRoot, frameType.ToString()); } public bool ChangeApplicationMode(WinoApplicationMode mode) => ExecuteOnNavigationThread(() => ChangeApplicationModeInternal(mode)); private bool ChangeApplicationModeInternal(WinoApplicationMode mode) { var coreFrame = GetCoreFrameInternal(NavigationReferenceFrame.ShellFrame); if (coreFrame == null) return false; // Update the application mode in state persistence service _statePersistanceService.ApplicationMode = mode; _statePersistanceService.CoreWindowTitle = mode == WinoApplicationMode.Calendar ? "Wino Calendar" : "Wino Mail"; var targetPageType = mode == WinoApplicationMode.Mail ? typeof(MailAppShell) : typeof(CalendarAppShell); var currentPageType = coreFrame.Content?.GetType(); var transitionInfo = GetNavigationTransitionInfo(NavigationTransitionType.DrillIn); // If already on the target page, do nothing if (currentPageType == targetPageType) return true; // Check if we can go back to the target page if (coreFrame.CanGoBack && coreFrame.BackStack.Count > 0) { var previousPage = coreFrame.BackStack[coreFrame.BackStack.Count - 1]; if (previousPage.SourcePageType == targetPageType) { coreFrame.GoBack(transitionInfo); return true; } } // Check if we can go forward to the target page if (coreFrame.CanGoForward && coreFrame.ForwardStack.Count > 0) { var nextPage = coreFrame.ForwardStack[coreFrame.ForwardStack.Count - 1]; if (nextPage.SourcePageType == targetPageType) { coreFrame.GoForward(); return true; } } // Navigate to the target page only if it's not in the navigation stack coreFrame.Navigate(targetPageType, null, transitionInfo); return true; } public bool Navigate(WinoPage page, object? parameter = null, NavigationReferenceFrame frame = NavigationReferenceFrame.InnerShellFrame, NavigationTransitionType transition = NavigationTransitionType.None) => ExecuteOnNavigationThread(() => NavigateInternal(page, parameter, frame, transition)); private bool NavigateInternal(WinoPage page, object? parameter = null, NavigationReferenceFrame frame = NavigationReferenceFrame.InnerShellFrame, NavigationTransitionType transition = NavigationTransitionType.None) { var pageType = GetPageType(page); if (pageType == null) return false; var currentApplicationMode = _statePersistanceService.ApplicationMode; if (currentApplicationMode == WinoApplicationMode.Calendar && IsMailOnlyPage(page)) { return false; } if (currentApplicationMode == WinoApplicationMode.Mail && IsCalendarOnlyPage(page)) { return false; } _statePersistanceService.IsReadingMail = _renderingPageTypes.Contains(page); _statePersistanceService.IsEventDetailsVisible = page == WinoPage.EventDetailsPage || page == WinoPage.CalendarEventComposePage; Frame? innerShellFrame = GetCoreFrameInternal(NavigationReferenceFrame.InnerShellFrame); if (innerShellFrame == null && frame == NavigationReferenceFrame.ShellFrame) { var requestedFrame = GetCoreFrameInternal(NavigationReferenceFrame.ShellFrame, WinoWindowKind.Welcome); if (requestedFrame == null) return false; return requestedFrame.Navigate(pageType, parameter, GetNavigationTransitionInfo(transition)); } if (innerShellFrame != null) { // Calendar navigations. if (currentApplicationMode == WinoApplicationMode.Calendar) { var currentFrameType = GetCurrentFrameType(innerShellFrame); if (page == WinoPage.CalendarPage && parameter is CalendarPageNavigationArgs calendarNavigationArgs) { var loadCalendarMessage = CreateLoadCalendarMessage(calendarNavigationArgs); // Date changes while CalendarPage is already active should not re-navigate the frame. if (currentFrameType == pageType) { WeakReferenceMessenger.Default.Send(loadCalendarMessage); return true; } // If CalendarPage is the previous page, reuse it instead of creating a second instance. var lastBackStackEntry = innerShellFrame.BackStack.Count > 0 ? innerShellFrame.BackStack[^1] : null; if (innerShellFrame.CanGoBack && lastBackStackEntry?.SourcePageType == pageType) { innerShellFrame.GoBack(); WeakReferenceMessenger.Default.Send(loadCalendarMessage); return true; } } return innerShellFrame.Navigate(pageType, parameter); } else { // Mail navigations. var currentFrameType = GetCurrentFrameType(innerShellFrame); bool isMailListingPageActive = currentFrameType != null && currentFrameType == typeof(MailListPage); // Active page is mail list page and we are refreshing the folder. if (isMailListingPageActive && currentFrameType == pageType && parameter is NavigateMailFolderEventArgs folderNavigationArgs) { // No need for new navigation, just refresh the folder. WeakReferenceMessenger.Default.Send(new ActiveMailFolderChangedEvent(folderNavigationArgs.BaseFolderMenuItem, folderNavigationArgs.FolderInitLoadAwaitTask)); WeakReferenceMessenger.Default.Send(new DisposeRenderingFrameRequested()); return true; } var transitionInfo = GetNavigationTransitionInfo(transition); // This page must be opened in the Frame placed in MailListingPage. if (isMailListingPageActive && frame == NavigationReferenceFrame.RenderingFrame) { var listingFrame = GetCoreFrameInternal(NavigationReferenceFrame.RenderingFrame); if (listingFrame == null) return false; // Active page is mail list page and we are opening a mail item. // No navigation needed, just refresh the rendered mail item. if (listingFrame.Content != null && listingFrame.Content.GetType() == GetPageType(WinoPage.MailRenderingPage) && parameter is MailItemViewModel mailItemViewModel && page != WinoPage.ComposePage) { WeakReferenceMessenger.Default.Send(new ReaderItemRefreshRequestedEvent(mailItemViewModel)); } else if (listingFrame.Content != null && listingFrame.Content.GetType() == GetPageType(WinoPage.ComposePage) && page == WinoPage.ComposePage && parameter is MailItemViewModel composeDraftViewModel) { // ComposePage is already active and we're switching to another draft. // Reuse existing ComposePage and WebView2 instead of navigating. WeakReferenceMessenger.Default.Send(new ReaderItemRefreshRequestedEvent(composeDraftViewModel)); } else if (listingFrame.Content != null && listingFrame.Content.GetType() == GetPageType(WinoPage.IdlePage) && pageType == typeof(IdlePage)) { // Idle -> Idle navigation. Ignore. return true; } else { listingFrame.Navigate(pageType, parameter, transitionInfo); } return true; } if ((currentFrameType != null && currentFrameType != pageType) || currentFrameType == null) { return innerShellFrame.Navigate(pageType, parameter, transitionInfo); } } } return false; } private static bool IsMailOnlyPage(WinoPage page) => MailOnlyPages.Contains(page); private static bool IsCalendarOnlyPage(WinoPage page) => CalendarOnlyPages.Contains(page); private static LoadCalendarMessage CreateLoadCalendarMessage(CalendarPageNavigationArgs args) { var targetDate = args.RequestDefaultNavigation ? DateTime.Now.Date : args.NavigationDate; var initiative = args.RequestDefaultNavigation ? CalendarInitInitiative.App : CalendarInitInitiative.User; return new LoadCalendarMessage(targetDate, initiative); } public void GoBack(Core.Domain.Enums.NavigationTransitionEffect slideEffect = Core.Domain.Enums.NavigationTransitionEffect.FromRight) => ExecuteOnNavigationThread(() => GoBackInternal(slideEffect)); private void GoBackInternal(Core.Domain.Enums.NavigationTransitionEffect slideEffect = Core.Domain.Enums.NavigationTransitionEffect.FromRight) { // Check if we're navigating within ManageAccountsPage (applies to both modes) // Check if we're navigating within SettingsPage (applies to both modes) if (_statePersistanceService.IsManageAccountsNavigating || _statePersistanceService.IsSettingsNavigating) { // Send message to ManageAccountsPage to go back within its AccountPagesFrame WeakReferenceMessenger.Default.Send(new BackBreadcrumNavigationRequested(slideEffect)); return; } var innerShellFrame = GetCoreFrameInternal(NavigationReferenceFrame.InnerShellFrame); if (_statePersistanceService.ApplicationMode == WinoApplicationMode.Calendar && innerShellFrame?.CanGoBack == true) { innerShellFrame.GoBack(); // Calendar mode: Navigate back from EventDetailsPage _statePersistanceService.IsEventDetailsVisible = false; } else { if (_statePersistanceService.IsReadingMail && _statePersistanceService.IsReaderNarrowed) { // Mail mode: Clear selections and dispose rendering frame _statePersistanceService.IsReadingMail = false; WeakReferenceMessenger.Default.Send(new ClearMailSelectionsRequested()); WeakReferenceMessenger.Default.Send(new DisposeRenderingFrameRequested()); } else if (innerShellFrame != null && innerShellFrame.CanGoBack) { innerShellFrame.GoBack(); } } } // Standalone EML viewer. //public void NavigateRendering(MimeMessageInformation mimeMessageInformation, NavigationTransitionType transition = NavigationTransitionType.None) //{ // if (mimeMessageInformation == null) // throw new ArgumentException("MimeMessage cannot be null."); // Navigate(WinoPage.MailRenderingPage, mimeMessageInformation, NavigationReferenceFrame.RenderingFrame, transition); //} //// Mail item view model clicked handler. //public void NavigateRendering(IMailItem mailItem, NavigationTransitionType transition = NavigationTransitionType.None) //{ // if (mailItem is MailItemViewModel mailItemViewModel) // Navigate(WinoPage.MailRenderingPage, mailItemViewModel, NavigationReferenceFrame.RenderingFrame, transition); // else // throw new ArgumentException("MailItem must be of type MailItemViewModel."); //} //public void NavigateFolder(NavigateMailFolderEventArgs args) // => Navigate(WinoPage.MailListPage, args, NavigationReferenceFrame.ShellFrame); }