MAUI中ViewModel应通过IWindow.GetNavigationProxy()或自定义INavigationService实现解耦导航,避免直接引用UI类型;Shell场景可使用Shell.Current.GoToAsync(),但需预注册路由并注意生命周期。

在 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 是目前最平衡的做法。










