【问题标题】:Navigating to a new page from the View Model in Windows Phone 8.1 universal app在 Windows Phone 8.1 通用应用程序中从视图模型导航到新页面
【发布时间】:2014-12-10 09:21:40
【问题描述】:

我正在开发一个 windows phone 8.1 通用应用程序,并希望找到处理页面导航的最佳方法,而无需在背后的代码中包含大量逻辑。我想尽可能保持视图中的代码整洁。响应按钮单击而导航到新页面的公认 MVVM 方式是什么?

我目前必须从 ViewModel 向视图发送 RelayComnmand 消息,其中包含要导航到的页面的详细信息。这意味着后面的代码必须按如下方式连接:

    public MainPage()
    {
      InitializeComponent();
      Messenger.Default.Register<OpenArticleMessage>(this, (article) => ReceiveOpenArticleMessage(article));
...
}

    private object ReceiveOpenArticleMessage(OpenArticleMessage article)
    {
     Frame.Navigate(typeof(ArticleView));
   }

这似乎不是最好的方法,尽管它确实有效。如何直接从 ViewModel 进行页面导航?我在我的项目中使用 MVVM-Light。

【问题讨论】:

标签: c# mvvm windows-phone-8.1 mvvm-light


【解决方案1】:

好的,我已经找到了这个问题的答案。进行了一些调查,但我最终找到了首选的 MVVM-Light 方法。无论如何,我不相信这个答案,只是在这里发布,以防人们正在寻找这个问题的答案。

如下创建一个INavigationService接口:

public interface INavigationService
{
    void Navigate(Type sourcePageType);
    void Navigate(Type sourcePageType, object parameter);
    void GoBack();
}

如下创建一个 NavigationService 类:

public class NavigationService : INavigationService
{
    public void Navigate(Type sourcePageType)
    {
        ((Frame)Window.Current.Content).Navigate(sourcePageType);
    }

    public void Navigate(Type sourcePageType, object parameter)
    {
        ((Frame)Window.Current.Content).Navigate(sourcePageType, parameter);
    }

    public void GoBack()
    {
        ((Frame)Window.Current.Content).GoBack();
    }
}

现在在 ViewModelLocator 中,像这样设置它:

    [System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Performance",
        "CA1822:MarkMembersAsStatic",
        Justification = "This non-static member is needed for data binding purposes.")]
    public MainViewModel Main
    {
        get
        {
            return ServiceLocator.Current.GetInstance<MainViewModel>();
        }
    }

    static ViewModelLocator()
    {
        ServiceLocator.SetLocatorProvider(() => SimpleIoc.Default);

        if (ViewModelBase.IsInDesignModeStatic)
        {
            SimpleIoc.Default.Register<INavigationService, Design.DesignNavigationService>();
        }
        else
        {
            SimpleIoc.Default.Register<INavigationService>(() => new NavigationService());
        }

        SimpleIoc.Default.Register<MainViewModel>();
    }

接下来为设计时设置导航服务,如下所示:

public class DesignNavigationService : INavigationService
{
    // This class doesn't perform navigation, in order
    // to avoid issues in the designer at design time.

    public void Navigate(Type sourcePageType)
    {
    }

    public void Navigate(Type sourcePageType, object parameter)
    {
    }

    public void GoBack()
    {
    }
}

我的 MainViewModel 构造函数如下:

   public MainViewModel(INavigationService navigationService)
    {
        _navigationService = navigationService;

        ...

现在您可以简单地使用它在您的视图模型中导航:

_navigationService.Navigate(typeof(WelcomeView));

有关原作者 Laurent Bugnion 的更多详细信息,请参阅本文和相关代码。 http://msdn.microsoft.com/en-us/magazine/jj651572.aspx

【讨论】:

  • 我们可以用什么方法来访问 backstack 条目?
  • 这里的问题是您的视图模型对视图类 (typeof(...)) 有依赖关系,这在严格的 MVVM 环境中应该避免...
  • 您应该使用导航服务中的字典创建视图到视图模型的映射。视图模型会告诉它想要导航的视图模型的类型,导航服务会查看视图模型类型(作为键)并找到相应的视图。这样,您的视图和视图模型就完全相互独立了。
【解决方案2】:

这里有一个新的更简单的实现:https://marcominerva.wordpress.com/2014/10/10/navigationservice-in-mvvm-light-v5/
首先我们创建NavigationServiceDialogService(用于页面导航参数):

public ViewModelLocator() {
    ServiceLocator.SetLocatorProvider(() => SimpleIoc.Default);

    var navigationService = this.CreateNavigationService();
    SimpleIoc.Default.Register<INavigationService>(() => navigationService);

    SimpleIoc.Default.Register<IDialogService, DialogService>();

    SimpleIoc.Default.Register<MainViewModel>();
    SimpleIoc.Default.Register<DetailsViewModel>();
}

private INavigationService CreateNavigationService() {
    var navigationService = new NavigationService();
    navigationService.Configure("Details", typeof(DetailsPage));
    // navigationService.Configure("key1", typeof(OtherPage1));
    // navigationService.Configure("key2", typeof(OtherPage2));

    return navigationService;
}

然后我们在您的ViewModel 中创建RelayCommandNavigationService,如下所示:

public class MainViewModel : ViewModelBase {
    private INavigationService _navigationService;
    public RelayCommand<Tuple<string, string>> DetailsCommand { get; set; }

    public MainViewModel(INavigationService navigationService) {
        this._navigationService = navigationService;
        DetailsCommand = new RelayCommand<Tuple<string, string>>((args) => NavigateTo(args));
    }

    public void NavigateTo(Tuple<string, string> args) {
        this._navigationService.NavigateTo(args.Item1, args.Item1);
    }

    public void ClickAndNavigate() {
        NavigateTo(new Tuple<string, string>("AdminPivotPage", "Test Params"));
    }
}

最后,我们可以像这样获取页面导航参数:

public sealed partial class DetailsPage : Page {
    // ... 
    protected override void OnNavigatedTo(NavigationEventArgs e) {
        var parameter = e.Parameter as string;  // "My data"
        base.OnNavigatedTo(e);
    }
}

但是要读取在 MVVM 模式中页面导航中传递的参数,您可以查看here

【讨论】:

【解决方案3】:

我同意上面的ricochete,它更简单,尽管我的直接实现与我在 Blend 中的设计数据绑定搞砸了。

我决定创建一个继承自 NavigationService 的类

public class NavigationServiceHelper : NavigationService
{
    public NavigationServiceHelper()
    {
        this.Configure("Page1", typeof(View.Page1));
        this.Configure("Page2", typeof(View.Page2));
    }
}

然后在 ViewModelLocator 中我以这种方式注册它

 SimpleIoc.Default.Register<INavigationService, NavigationServiceHelper>();

我的设计视图数据绑定再次起作用。如果有人可以解释为什么设计数据在上面的ricochete 中不起作用,请这样做。谢谢!

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多