Caliburn.Micro 3.0 相当于 Xamarin.Forms Navigation.PushModalAsync
Caliburn.Micro 3.0 equivalent to Xamarin.Forms Navigation.PushModalAsync
Caliburn.Micro 3.0(和 Caliburn.Micro.Xamarin.Forms)是否在 Xamarin.Forms 中实现 mimic/support Navigation.PushModalAsync
的功能?
没有。它不是内置的,但很容易增强它。通常,MvvM 框架由 ViewModels 导航。 Caliburn 正在遵循这种模式。所以它需要某种导航服务。此导航服务负责为 ViewModel 创建视图并调用视图框架(Xamarin.Froms 在我们的例子中)特定的导航功能。 NavigationPageAdapter
就是我们要找的东西。现在让我们加强它。
public interface IModalNavigationService : INavigationService
{
Task NavigateModalToViewModelAsync<TViewModel>(object parameter = null, bool animated = true);
// TODO: add more functions for closing
}
public class ModalNavigationPageAdapter : NavigationPageAdapter, IModalNavigationService
{
private readonly NavigationPage _navigationPage;
public ModalNavigationPageAdapter(NavigationPage navigationPage) : base(navigationPage)
{
_navigationPage = navigationPage;
}
public async Task NavigateModalToViewModelAsync<TViewModel>(object parameter = null, bool animated = true)
{
var view = ViewLocator.LocateForModelType(typeof(TViewModel), null, null);
await PushModalAsync(view, parameter, animated);
}
private Task PushModalAsync(Element view, object parameter, bool animated)
{
var page = view as Page;
if (page == null)
throw new NotSupportedException(String.Format("{0} does not inherit from {1}.", view.GetType(), typeof(Page)));
var viewModel = ViewModelLocator.LocateForView(view);
if (viewModel != null)
{
TryInjectParameters(viewModel, parameter);
ViewModelBinder.Bind(viewModel, view, null);
}
page.Appearing += (s, e) => ActivateView(page);
page.Disappearing += (s, e) => DeactivateView(page);
return _navigationPage.Navigation.PushModalAsync(page, animated);
}
private static void DeactivateView(BindableObject view)
{
if (view == null)
return;
var deactivate = view.BindingContext as IDeactivate;
if (deactivate != null)
{
deactivate.Deactivate(false);
}
}
private static void ActivateView(BindableObject view)
{
if (view == null)
return;
var activator = view.BindingContext as IActivate;
if (activator != null)
{
activator.Activate();
}
}
}
我们刚刚声明了扩展 INavigationService
的接口 IModalNavigationService
并在我们的 ModalNavigationPageAdapter
中实现了它。不幸的是,Caliburn 将很多函数设为私有,因此我们必须将它们复制到我们继承的版本中。
在 Caliburn 中,您可以通过 navigationservice.For<VM>().Navigate()
导航。我们想遵循这种风格,所以我们必须在扩展方法中实现类似 navigationservice.ModalFor<VM>().Navigate()
的东西。
public static class ModalNavigationExtensions
{
public static ModalNavigateHelper<TViewModel> ModalFor<TViewModel>(this IModalNavigationService navigationService)
{
return new ModalNavigateHelper<TViewModel>().AttachTo(navigationService);
}
}
此方法 returns ModalNavigateHelper
简化了我们导航服务的使用(类似于 Caliburn 的 NavigateHelper
)。这几乎是一个副本,但对于 IModalNavigationService
。
public class ModalNavigateHelper<TViewModel>
{
readonly Dictionary<string, object> parameters = new Dictionary<string, object>();
IModalNavigationService navigationService;
public ModalNavigateHelper<TViewModel> WithParam<TValue>(Expression<Func<TViewModel, TValue>> property, TValue value)
{
if (value is ValueType || !ReferenceEquals(null, value))
{
parameters[property.GetMemberInfo().Name] = value;
}
return this;
}
public ModalNavigateHelper<TViewModel> AttachTo(IModalNavigationService navigationService)
{
this.navigationService = navigationService;
return this;
}
public void Navigate(bool animated = true)
{
if (navigationService == null)
{
throw new InvalidOperationException("Cannot navigate without attaching an INavigationService. Call AttachTo first.");
}
navigationService.NavigateModalToViewModelAsync<TViewModel>(parameters, animated);
}
}
最后但同样重要的是,我们必须使用闪亮的新导航服务而不是旧导航服务。 App
class 正在将 INavigationService
的 NavigationPageAdapter
注册为 PrepareViewFirst
中的单例。我们必须改变它如下
public class App : FormsApplication
{
private readonly SimpleContainer container;
public App(SimpleContainer container)
{
this.container = container;
container
.PerRequest<LoginViewModel>()
.PerRequest<FeaturesViewModel>();
Initialize();
DisplayRootView<LoginView>();
}
protected override void PrepareViewFirst(NavigationPage navigationPage)
{
var navigationService = new ModalNavigationPageAdapter(navigationPage);
container.Instance<INavigationService>(navigationService);
container.Instance<IModalNavigationService>(navigationService);
}
}
我们正在为 INavigationService
和 IModalNavigationService
注册我们的导航服务。
如评论中所见,必须自己实现调用PopModalAsync
的close函数
Caliburn.Micro 3.0(和 Caliburn.Micro.Xamarin.Forms)是否在 Xamarin.Forms 中实现 mimic/support Navigation.PushModalAsync
的功能?
没有。它不是内置的,但很容易增强它。通常,MvvM 框架由 ViewModels 导航。 Caliburn 正在遵循这种模式。所以它需要某种导航服务。此导航服务负责为 ViewModel 创建视图并调用视图框架(Xamarin.Froms 在我们的例子中)特定的导航功能。 NavigationPageAdapter
就是我们要找的东西。现在让我们加强它。
public interface IModalNavigationService : INavigationService
{
Task NavigateModalToViewModelAsync<TViewModel>(object parameter = null, bool animated = true);
// TODO: add more functions for closing
}
public class ModalNavigationPageAdapter : NavigationPageAdapter, IModalNavigationService
{
private readonly NavigationPage _navigationPage;
public ModalNavigationPageAdapter(NavigationPage navigationPage) : base(navigationPage)
{
_navigationPage = navigationPage;
}
public async Task NavigateModalToViewModelAsync<TViewModel>(object parameter = null, bool animated = true)
{
var view = ViewLocator.LocateForModelType(typeof(TViewModel), null, null);
await PushModalAsync(view, parameter, animated);
}
private Task PushModalAsync(Element view, object parameter, bool animated)
{
var page = view as Page;
if (page == null)
throw new NotSupportedException(String.Format("{0} does not inherit from {1}.", view.GetType(), typeof(Page)));
var viewModel = ViewModelLocator.LocateForView(view);
if (viewModel != null)
{
TryInjectParameters(viewModel, parameter);
ViewModelBinder.Bind(viewModel, view, null);
}
page.Appearing += (s, e) => ActivateView(page);
page.Disappearing += (s, e) => DeactivateView(page);
return _navigationPage.Navigation.PushModalAsync(page, animated);
}
private static void DeactivateView(BindableObject view)
{
if (view == null)
return;
var deactivate = view.BindingContext as IDeactivate;
if (deactivate != null)
{
deactivate.Deactivate(false);
}
}
private static void ActivateView(BindableObject view)
{
if (view == null)
return;
var activator = view.BindingContext as IActivate;
if (activator != null)
{
activator.Activate();
}
}
}
我们刚刚声明了扩展 INavigationService
的接口 IModalNavigationService
并在我们的 ModalNavigationPageAdapter
中实现了它。不幸的是,Caliburn 将很多函数设为私有,因此我们必须将它们复制到我们继承的版本中。
在 Caliburn 中,您可以通过 navigationservice.For<VM>().Navigate()
导航。我们想遵循这种风格,所以我们必须在扩展方法中实现类似 navigationservice.ModalFor<VM>().Navigate()
的东西。
public static class ModalNavigationExtensions
{
public static ModalNavigateHelper<TViewModel> ModalFor<TViewModel>(this IModalNavigationService navigationService)
{
return new ModalNavigateHelper<TViewModel>().AttachTo(navigationService);
}
}
此方法 returns ModalNavigateHelper
简化了我们导航服务的使用(类似于 Caliburn 的 NavigateHelper
)。这几乎是一个副本,但对于 IModalNavigationService
。
public class ModalNavigateHelper<TViewModel>
{
readonly Dictionary<string, object> parameters = new Dictionary<string, object>();
IModalNavigationService navigationService;
public ModalNavigateHelper<TViewModel> WithParam<TValue>(Expression<Func<TViewModel, TValue>> property, TValue value)
{
if (value is ValueType || !ReferenceEquals(null, value))
{
parameters[property.GetMemberInfo().Name] = value;
}
return this;
}
public ModalNavigateHelper<TViewModel> AttachTo(IModalNavigationService navigationService)
{
this.navigationService = navigationService;
return this;
}
public void Navigate(bool animated = true)
{
if (navigationService == null)
{
throw new InvalidOperationException("Cannot navigate without attaching an INavigationService. Call AttachTo first.");
}
navigationService.NavigateModalToViewModelAsync<TViewModel>(parameters, animated);
}
}
最后但同样重要的是,我们必须使用闪亮的新导航服务而不是旧导航服务。 App
class 正在将 INavigationService
的 NavigationPageAdapter
注册为 PrepareViewFirst
中的单例。我们必须改变它如下
public class App : FormsApplication
{
private readonly SimpleContainer container;
public App(SimpleContainer container)
{
this.container = container;
container
.PerRequest<LoginViewModel>()
.PerRequest<FeaturesViewModel>();
Initialize();
DisplayRootView<LoginView>();
}
protected override void PrepareViewFirst(NavigationPage navigationPage)
{
var navigationService = new ModalNavigationPageAdapter(navigationPage);
container.Instance<INavigationService>(navigationService);
container.Instance<IModalNavigationService>(navigationService);
}
}
我们正在为 INavigationService
和 IModalNavigationService
注册我们的导航服务。
如评论中所见,必须自己实现调用PopModalAsync
的close函数