1

我已经开始使用微软的新 UI 技术 WinUI 3。

我的问题类似于这个WinUI 3.0 Desktop - C# Page Navigation,但我特别想知道导航到现有页面实例。

我的应用程序的主菜单包含一个NavigationView控件。

备注:我安装了这个示例解决方案(“xaml-controls-gallery”)以了解它的基本概念。

目前我在其中有 3 个NavigationViewItem实例:

    <NavigationView x:Name="navigationView" PaneDisplayMode="Top" 
                    SelectionFollowsFocus="Enabled" IsBackButtonVisible="Collapsed" ItemInvoked="NavigationView_OnItemInvoked"
    >

        <NavigationView.MenuItems>
            <NavigationViewItem Icon="Play" Content="Menu 1" x:Name="Menu1Item" Tag="Page1" />
            <NavigationViewItem Icon="Filter" Content="Menu 2" x:Name="Menu2Item" Tag="Page2"/>
            <NavigationViewItem Icon="Refresh" Content="Menu 3" x:Name="Menu3Item" Tag="Page3" />
        </NavigationView.MenuItems>

        <!-- The section will contain the content for selected navigation item -->
        <Grid Padding="20">
            <Frame x:Name="rootFrame" Navigated="RootFrame_OnNavigated"/>
        </Grid>

    </NavigationView>

属于相应菜单项的内容应显示在框架“rootFrame”中。对于内容,我创建了 Page 类:Page1、Page2 和 Page3。

在文件后面的代码中,我处理 NavigationView 的“ItemInvoked”事件。我检查 NavigationItem 的“标签”并设置要导航到的页面的类型。

    private void NavigationView_OnItemInvoked(Microsoft.UI.Xaml.Controls.NavigationView sender, Microsoft.UI.Xaml.Controls.NavigationViewItemInvokedEventArgs args)
    {
        FrameNavigationOptions navOptions = new FrameNavigationOptions();
        navOptions.TransitionInfoOverride = args.RecommendedNavigationTransitionInfo;

        string navItemTag = args.InvokedItemContainer.Tag.ToString();

        Type pageType = null;
        if (navItemTag == "Page1")
        {
            pageType = typeof(Page1);
        }
        else if (navItemTag == "Page2")
        {
            pageType = typeof(Page2);
        }
        else if (navItemTag == "Page3")
        {
            pageType = typeof(Page3);
        }

        if (pageType == null)
        {
            return;
        }

        rootFrame.NavigateToType(pageType, null, navOptions);
    }

但是,在Frame实例上调用“NavigateToType”总是会创建页面类的新实例。我想要的是不要放弃用户也对 UI 进行了一些更改的页面状态。所以我想导航到已经存在的页面实例。

我不知道如何做到这一点。

因此,在页面的“OnNavigatedTo”事件中, “e”参数的NavigationMode始终为 NavigationMode.New 而不是 NavigationMode.Refresh。这是“Page1”的示例:

public sealed partial class Page1 : Page
{
    public Page1()
    {
        this.InitializeComponent();
        
        ViewModel = new Page1ViewModel();
        // Load data (expensive operation)
    }

    public Page1ViewModel ViewModel { get; set; }


    protected override void OnNavigatedTo(NavigationEventArgs e)
    {
        base.OnNavigatedTo(e);

        // e.NavigationMode is always NavigationMode.New 
        // How do we get NavigationMode.Refresh?
    }
}

有人知道如何导航到现有页面实例吗?

作为一种解决方法,我尝试直接交换框架的内容:

rootFrame.Content = instanceOfPage;

但我真的不想这样做,因为这会绕过基于框架的导航,并且在导航到其他页面时也不会显示动画。

似乎在 WPF 和“旧”.NET 框架中,有一个类NavigationWindow用于此目的,但在 WinUi 中(尚不存在)。

4

1 回答 1

2

将页面的NavigationCacheMode属性设置为RequiredEnabled

<Page
    x:Class="WinUI3App.Page1"
    NavigationCacheMode="Enabled"
    ...

另见这篇文章:实现两个页面之间的导航 - 4. 缓存一个页面

于 2021-09-22T16:22:30.813 回答