685 lines
28 KiB
C#
685 lines
28 KiB
C#
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.Core.Domain.Models.Settings;
|
|
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;
|
|
using Wino.Mail.WinUI.Views.Calendar;
|
|
using Wino.Messaging.Client.Calendar;
|
|
using Wino.Messaging.Client.Mails;
|
|
using Wino.Messaging.Client.Navigation;
|
|
using Wino.Views;
|
|
using Wino.Views.Account;
|
|
using Wino.Views.Mail;
|
|
using Wino.Views.Settings;
|
|
using Microsoft.UI.Xaml.Media.Animation;
|
|
|
|
namespace Wino.Services;
|
|
|
|
public class NavigationService : NavigationServiceBase, INavigationService
|
|
{
|
|
private readonly IStatePersistanceService _statePersistanceService;
|
|
private readonly IDispatcher _dispatcher;
|
|
private readonly IWinoWindowManager _windowManager;
|
|
private NavigationTransitionInfo? _pendingInnerShellTransition;
|
|
|
|
private WinoPage[] _renderingPageTypes = new WinoPage[]
|
|
{
|
|
WinoPage.MailRenderingPage,
|
|
WinoPage.ComposePage
|
|
};
|
|
|
|
private static readonly WinoPage[] MailOnlyPages =
|
|
[
|
|
WinoPage.MailListPage,
|
|
WinoPage.MailRenderingPage,
|
|
WinoPage.ComposePage,
|
|
WinoPage.IdlePage,
|
|
WinoPage.WelcomePageV2,
|
|
WinoPage.WelcomeHostPage,
|
|
WinoPage.ProviderSelectionPage,
|
|
WinoPage.AccountSetupProgressPage,
|
|
WinoPage.SpecialImapCredentialsPage
|
|
];
|
|
|
|
private static readonly WinoPage[] CalendarOnlyPages =
|
|
[
|
|
WinoPage.CalendarPage,
|
|
WinoPage.EventDetailsPage,
|
|
WinoPage.CalendarEventComposePage
|
|
];
|
|
|
|
private static readonly WinoPage[] ContactsOnlyPages =
|
|
[
|
|
WinoPage.ContactsPage
|
|
];
|
|
|
|
private static readonly WinoPage[] SettingsOnlyPages =
|
|
[
|
|
WinoPage.SettingsPage,
|
|
WinoPage.SettingOptionsPage,
|
|
WinoPage.ManageAccountsPage,
|
|
WinoPage.AccountManagementPage,
|
|
WinoPage.AccountDetailsPage,
|
|
WinoPage.MergedAccountDetailsPage,
|
|
WinoPage.SignatureManagementPage,
|
|
WinoPage.AboutPage,
|
|
WinoPage.PersonalizationPage,
|
|
WinoPage.MessageListPage,
|
|
WinoPage.ReadComposePanePage,
|
|
WinoPage.AppPreferencesPage,
|
|
WinoPage.AliasManagementPage,
|
|
WinoPage.ImapCalDavSettingsPage,
|
|
WinoPage.KeyboardShortcutsPage,
|
|
WinoPage.SignatureAndEncryptionPage,
|
|
WinoPage.EmailTemplatesPage,
|
|
WinoPage.CreateEmailTemplatePage,
|
|
WinoPage.StoragePage,
|
|
WinoPage.WinoAccountManagementPage,
|
|
WinoPage.CalendarSettingsPage,
|
|
WinoPage.CalendarRenderingSettingsPage,
|
|
WinoPage.CalendarNotificationSettingsPage,
|
|
WinoPage.CalendarPreferenceSettingsPage,
|
|
WinoPage.CalendarAccountSettingsPage
|
|
];
|
|
|
|
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<T>(Func<T> 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(AccountManagementPage),
|
|
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.WelcomePageV2 => typeof(WelcomePageV2),
|
|
WinoPage.SettingOptionsPage => typeof(SettingOptionsPage),
|
|
WinoPage.AppPreferencesPage => typeof(AppPreferencesPage),
|
|
WinoPage.AliasManagementPage => typeof(AliasManagementPage),
|
|
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.WinoAccountManagementPage => typeof(WinoAccountManagementPage),
|
|
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(CalendarPreferenceSettingsPage),
|
|
WinoPage.CalendarRenderingSettingsPage => typeof(CalendarRenderingSettingsPage),
|
|
WinoPage.CalendarNotificationSettingsPage => typeof(CalendarNotificationSettingsPage),
|
|
WinoPage.CalendarPreferenceSettingsPage => typeof(CalendarPreferenceSettingsPage),
|
|
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<Frame>(contentRoot, frameType.ToString());
|
|
}
|
|
|
|
public bool ChangeApplicationMode(WinoApplicationMode mode)
|
|
=> ExecuteOnNavigationThread(() => ChangeApplicationModeInternal(mode));
|
|
|
|
public bool ChangeApplicationMode(WinoApplicationMode mode, ShellModeActivationContext activationContext)
|
|
=> ExecuteOnNavigationThread(() => ChangeApplicationModeInternal(mode, activationContext));
|
|
|
|
public bool CanGoBack()
|
|
=> ExecuteOnNavigationThread(CanGoBackInternal);
|
|
|
|
private bool ChangeApplicationModeInternal(WinoApplicationMode mode, ShellModeActivationContext? activationContext = null)
|
|
{
|
|
var coreFrame = GetCoreFrameInternal(NavigationReferenceFrame.ShellFrame);
|
|
|
|
if (coreFrame == null) return false;
|
|
|
|
var currentMode = _statePersistanceService.ApplicationMode;
|
|
var isInitialShellNavigation = coreFrame.Content is not IShellHost;
|
|
|
|
// Update the application mode in state persistence service
|
|
_statePersistanceService.ApplicationMode = mode;
|
|
_statePersistanceService.AppModeTitle = GetApplicationModeTitle(mode);
|
|
|
|
if (coreFrame.Content is IShellHost activeShell && activeShell.HasShellContent && currentMode == mode)
|
|
return true;
|
|
|
|
_pendingInnerShellTransition = isInitialShellNavigation
|
|
? null
|
|
: GetApplicationModeTransitionInfo(currentMode, mode);
|
|
|
|
if (coreFrame.Content is not IShellHost)
|
|
{
|
|
coreFrame.BackStack.Clear();
|
|
coreFrame.ForwardStack.Clear();
|
|
coreFrame.Navigate(typeof(WinoAppShell), null, new SuppressNavigationTransitionInfo());
|
|
}
|
|
|
|
if (coreFrame.Content is IShellHost shell)
|
|
{
|
|
shell.ActivateMode(mode, new ShellModeActivationContext
|
|
{
|
|
IsInitialActivation = isInitialShellNavigation,
|
|
SuppressStartupFlows = activationContext?.SuppressStartupFlows ?? false,
|
|
Parameter = activationContext?.Parameter
|
|
});
|
|
|
|
ResetCurrentModeBackStackState();
|
|
return true;
|
|
}
|
|
|
|
_pendingInnerShellTransition = null;
|
|
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)
|
|
{
|
|
if (TryGetSettingsActivationTarget(page, parameter, out var settingsTarget))
|
|
{
|
|
if (_statePersistanceService.ApplicationMode != WinoApplicationMode.Settings)
|
|
{
|
|
return ChangeApplicationModeInternal(WinoApplicationMode.Settings, new ShellModeActivationContext
|
|
{
|
|
Parameter = settingsTarget
|
|
});
|
|
}
|
|
|
|
page = WinoPage.SettingsPage;
|
|
parameter = settingsTarget;
|
|
}
|
|
|
|
var pageType = GetPageType(page);
|
|
if (pageType == null) return false;
|
|
|
|
var currentApplicationMode = _statePersistanceService.ApplicationMode;
|
|
|
|
if (!IsPageAllowedInMode(currentApplicationMode, 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)
|
|
{
|
|
PruneInnerShellBackStackForMode(innerShellFrame, currentApplicationMode);
|
|
|
|
// 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 NavigateInnerShellFrame(innerShellFrame, pageType, parameter, transition);
|
|
}
|
|
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;
|
|
}
|
|
|
|
// 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;
|
|
|
|
var transitionInfo = GetNavigationTransitionInfo(transition);
|
|
|
|
// 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)
|
|
{
|
|
if (listingFrame.Content is MailRenderingPage renderingPage)
|
|
{
|
|
_ = renderingPage.RefreshMailItemAsync(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.
|
|
if (listingFrame.Content is ComposePage composePage)
|
|
{
|
|
_ = composePage.RefreshDraftAsync(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 NavigateInnerShellFrame(innerShellFrame, pageType, parameter, transition);
|
|
}
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
private static bool IsMailOnlyPage(WinoPage page)
|
|
=> MailOnlyPages.Contains(page);
|
|
|
|
private static bool IsCalendarOnlyPage(WinoPage page)
|
|
=> CalendarOnlyPages.Contains(page);
|
|
|
|
private static bool IsContactsOnlyPage(WinoPage page)
|
|
=> ContactsOnlyPages.Contains(page);
|
|
|
|
private static bool IsSettingsOnlyPage(WinoPage page)
|
|
=> SettingsOnlyPages.Contains(page);
|
|
|
|
private static bool IsPageAllowedInMode(WinoApplicationMode mode, WinoPage page)
|
|
=> mode switch
|
|
{
|
|
WinoApplicationMode.Mail => !IsCalendarOnlyPage(page) && !IsContactsOnlyPage(page) && !IsSettingsOnlyPage(page),
|
|
WinoApplicationMode.Calendar => !IsMailOnlyPage(page) && !IsContactsOnlyPage(page) && !IsSettingsOnlyPage(page),
|
|
WinoApplicationMode.Contacts => !IsMailOnlyPage(page) && !IsCalendarOnlyPage(page) && !IsSettingsOnlyPage(page),
|
|
WinoApplicationMode.Settings => IsSettingsOnlyPage(page),
|
|
_ => true
|
|
};
|
|
|
|
private static string GetApplicationModeTitle(WinoApplicationMode mode)
|
|
=> mode switch
|
|
{
|
|
WinoApplicationMode.Calendar => "Wino Calendar",
|
|
WinoApplicationMode.Contacts => "Wino Contacts",
|
|
WinoApplicationMode.Settings => "Wino Settings",
|
|
_ => "Wino Mail"
|
|
};
|
|
|
|
private static NavigationTransitionInfo GetApplicationModeTransitionInfo(WinoApplicationMode currentMode, WinoApplicationMode targetMode)
|
|
{
|
|
var slideEffect = IsNextMode(currentMode, targetMode)
|
|
? SlideNavigationTransitionEffect.FromRight
|
|
: SlideNavigationTransitionEffect.FromLeft;
|
|
|
|
return new SlideNavigationTransitionInfo
|
|
{
|
|
Effect = slideEffect
|
|
};
|
|
}
|
|
|
|
private static bool IsNextMode(WinoApplicationMode currentMode, WinoApplicationMode targetMode)
|
|
=> currentMode switch
|
|
{
|
|
WinoApplicationMode.Mail => targetMode == WinoApplicationMode.Calendar,
|
|
WinoApplicationMode.Calendar => targetMode == WinoApplicationMode.Contacts,
|
|
WinoApplicationMode.Contacts => targetMode == WinoApplicationMode.Settings,
|
|
WinoApplicationMode.Settings => targetMode == WinoApplicationMode.Mail,
|
|
_ => false
|
|
};
|
|
|
|
private static bool TryGetSettingsActivationTarget(WinoPage page, object? parameter, out WinoPage targetPage)
|
|
{
|
|
targetPage = WinoPage.SettingOptionsPage;
|
|
|
|
if (page == WinoPage.SettingsPage)
|
|
{
|
|
targetPage = parameter as WinoPage? ?? WinoPage.SettingOptionsPage;
|
|
return true;
|
|
}
|
|
|
|
if (!IsSettingsOnlyPage(page))
|
|
return false;
|
|
|
|
targetPage = SettingsNavigationInfoProvider.GetRootPage(page);
|
|
return true;
|
|
}
|
|
|
|
private LoadCalendarMessage CreateLoadCalendarMessage(CalendarPageNavigationArgs args)
|
|
{
|
|
var targetDate = args.RequestDefaultNavigation
|
|
? DateOnly.FromDateTime(DateTime.Now.Date)
|
|
: DateOnly.FromDateTime(args.NavigationDate.Date);
|
|
|
|
var displayRequest = new CalendarDisplayRequest(_statePersistanceService.CalendarDisplayType, targetDate);
|
|
return new LoadCalendarMessage(displayRequest, args.ForceReload);
|
|
}
|
|
|
|
private bool NavigateInnerShellFrame(Frame frame, Type pageType, object? parameter, NavigationTransitionType transition)
|
|
{
|
|
var transitionInfo = ConsumeInnerShellTransitionOrDefault(transition);
|
|
var navigationResult = frame.Navigate(pageType, parameter, transitionInfo);
|
|
|
|
if (navigationResult)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
return navigationResult;
|
|
}
|
|
|
|
private NavigationTransitionInfo ConsumeInnerShellTransitionOrDefault(NavigationTransitionType transition)
|
|
{
|
|
if (_pendingInnerShellTransition != null)
|
|
{
|
|
var transitionInfo = _pendingInnerShellTransition;
|
|
_pendingInnerShellTransition = null;
|
|
return transitionInfo;
|
|
}
|
|
|
|
return GetNavigationTransitionInfo(transition);
|
|
}
|
|
|
|
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)
|
|
{
|
|
var innerShellFrame = GetCoreFrameInternal(NavigationReferenceFrame.InnerShellFrame);
|
|
var currentApplicationMode = _statePersistanceService.ApplicationMode;
|
|
|
|
if (currentApplicationMode == WinoApplicationMode.Settings &&
|
|
innerShellFrame?.Content is SettingsPage settingsPage)
|
|
{
|
|
if (settingsPage.CanNavigateBack)
|
|
{
|
|
WeakReferenceMessenger.Default.Send(new BackBreadcrumNavigationRequested(slideEffect));
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
if (innerShellFrame != null)
|
|
{
|
|
PruneInnerShellBackStackForMode(innerShellFrame, currentApplicationMode);
|
|
}
|
|
|
|
if (currentApplicationMode == WinoApplicationMode.Calendar)
|
|
{
|
|
if (innerShellFrame?.CanGoBack == true)
|
|
{
|
|
innerShellFrame.GoBack();
|
|
}
|
|
else if (innerShellFrame != null && innerShellFrame.Content?.GetType() != typeof(CalendarPage))
|
|
{
|
|
NavigateToCalendarRoot(innerShellFrame);
|
|
}
|
|
|
|
// Calendar mode: Navigate back from EventDetailsPage
|
|
_statePersistanceService.IsEventDetailsVisible = false;
|
|
}
|
|
else if (currentApplicationMode == WinoApplicationMode.Mail)
|
|
{
|
|
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());
|
|
}
|
|
}
|
|
}
|
|
|
|
private void ResetCurrentModeBackStackState()
|
|
{
|
|
var innerShellFrame = GetCoreFrameInternal(NavigationReferenceFrame.InnerShellFrame);
|
|
|
|
if (innerShellFrame != null)
|
|
{
|
|
innerShellFrame.BackStack.Clear();
|
|
innerShellFrame.ForwardStack.Clear();
|
|
}
|
|
}
|
|
|
|
private void PruneInnerShellBackStackForMode(Frame frame, WinoApplicationMode mode)
|
|
{
|
|
for (int i = frame.BackStack.Count - 1; i >= 0; i--)
|
|
{
|
|
var backStackEntry = frame.BackStack[i];
|
|
|
|
if (!IsPageTypeAllowedInMode(mode, backStackEntry.SourcePageType))
|
|
{
|
|
frame.BackStack.RemoveAt(i);
|
|
}
|
|
}
|
|
}
|
|
|
|
private bool IsPageTypeAllowedInMode(WinoApplicationMode mode, Type? pageType)
|
|
{
|
|
if (pageType == null)
|
|
return false;
|
|
|
|
foreach (var page in Enum.GetValues<WinoPage>())
|
|
{
|
|
if (page == WinoPage.None)
|
|
continue;
|
|
|
|
if (GetPageType(page) == pageType)
|
|
return IsPageAllowedInMode(mode, page);
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
private void NavigateToCalendarRoot(Frame frame)
|
|
{
|
|
if (!frame.Navigate(typeof(CalendarPage), new CalendarPageNavigationArgs
|
|
{
|
|
RequestDefaultNavigation = true
|
|
}, GetNavigationTransitionInfo(NavigationTransitionType.None)))
|
|
{
|
|
return;
|
|
}
|
|
|
|
frame.BackStack.Clear();
|
|
frame.ForwardStack.Clear();
|
|
}
|
|
|
|
private bool CanGoBackInternal()
|
|
{
|
|
var innerShellFrame = GetCoreFrameInternal(NavigationReferenceFrame.InnerShellFrame);
|
|
|
|
return _statePersistanceService.ApplicationMode switch
|
|
{
|
|
WinoApplicationMode.Mail => _statePersistanceService.IsReadingMail && _statePersistanceService.IsReaderNarrowed,
|
|
WinoApplicationMode.Settings => innerShellFrame?.Content is SettingsPage settingsPage && settingsPage.CanNavigateBack,
|
|
WinoApplicationMode.Calendar or WinoApplicationMode.Contacts => HasModeScopedBackStack(innerShellFrame, _statePersistanceService.ApplicationMode),
|
|
_ => false
|
|
};
|
|
}
|
|
|
|
private bool HasModeScopedBackStack(Frame? innerShellFrame, WinoApplicationMode mode)
|
|
{
|
|
if (innerShellFrame == null || innerShellFrame.BackStack.Count == 0)
|
|
return false;
|
|
|
|
for (int i = innerShellFrame.BackStack.Count - 1; i >= 0; i--)
|
|
{
|
|
if (IsPageTypeAllowedInMode(mode, innerShellFrame.BackStack[i].SourcePageType))
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
// 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);
|
|
}
|