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