在 MAUI 中,ViewModel 本身不直接持有导航逻辑,但可以通过
IWindow或
INavigation的注入方式实现解耦导航。官方推荐做法是:**在 ViewModel 中触发导航请求,由 View 层(或服务层)执行实际跳转**,避免 ViewModel 引用 UI 类型(如 Page),保持可测试性。
使用 IServiceProvider 获取 NavigationService(推荐)
MAUI 默认提供
INavigation实例,但仅在
ContentPage等页面中可用。要在 ViewModel 中使用,需通过依赖注入获取当前上下文的导航服务: 在
App.xaml.cs的
ConfigureServices中注册导航服务(如自定义
NavigationService) 或在 ViewModel 构造函数中注入
IWindow(.NET 8+ 支持),再通过
window.GetNavigationProxy()获取
INavigation示例(.NET 8+):
public MyViewModel(IWindow window) => _navigation = window.GetNavigationProxy();
await _navigation.PushAsync(new DetailPage());
自定义 NavigationService 封装(更灵活、可测试)
创建一个不依赖 UI 的导航服务,内部代理到当前窗口的
INavigation: 定义接口
INavigationService,含
GoToAsync(string route)、
GoBackAsync()等方法 实现类中通过
Application.Current?.Windows.FirstOrDefault(w => w.IsActive)找到活跃窗口,再调用其
GetNavigationProxy()在 ViewModel 中注入该服务,即可安全调用导航,且单元测试时可 Mock
配合 Shell 路由时,在 ViewModel 中使用 Shell.Current
若项目使用 MAUI Shell,可直接在 ViewModel 中访问全局
Shell.Current(需注意线程安全和生命周期):
await Shell.Current.GoToAsync("//home/details");
确保路由已通过 Routing.RegisterRoute()预注册 不推荐在非 Shell 场景或复杂导航流中强依赖
Shell.Current,因其耦合 Shell 生命周期
基本上就这些。关键不是“能不能在 ViewModel 导航”,而是“怎么导得干净、可测、不破环分层”。用
IWindow.GetNavigationProxy()或封装好的
INavigationService是目前最平衡的做法。
