xamarin - 如何使用 MvvmCross 6 在 Xamarin.iOS 中的代码中正确创建根视图控制器?
问题描述
我正在尝试用代码编写 Xamarin.iOS UI,我正在关注这个 MS 教程。不幸的是,我无法将我的MvxTabBarViewController
( MainViewController
) 设置为根控制器。使用 NullReferenceException 实例化控制器结果。
在 MvvmCross 中将控制器设置为根控制器的正确方法是什么?
我的 AppDelegate 如下:
[Register(nameof(AppDelegate))]
public class AppDelegate : MvxApplicationDelegate<Setup, App>
{
public override void FinishedLaunching(UIApplication application)
{
base.FinishedLaunching(application);
}
public override bool FinishedLaunching(UIApplication application, NSDictionary launchOptions)
{
Window = new UIWindow(UIScreen.MainScreen.Bounds);
var controller = new MainViewController();
Window.RootViewController = controller;
Window.MakeKeyAndVisible();
Debug.WriteLine("Set root view to MainViewController");
return true;
}
}
并且视图控制器是根:
[MvxRootPresentation]
public partial class MainViewController : MvxTabBarViewController<MainViewModel>
{
private UILabel label;
//public MainViewController(IntPtr handle) : base(handle)
//{
//}
public override void ViewDidLoad()
{
base.ViewDidLoad();
Debug.WriteLine("Main View Controller loaded");
AddLabel();
}
private void AddLabel()
{
label = new UILabel
{
Text = "testing",
TranslatesAutoresizingMaskIntoConstraints = false
};
var lblConstraints = new[]
{
label.LeadingAnchor.ConstraintEqualTo(anchor:this.View.SafeAreaLayoutGuide.LeadingAnchor, constant:20.0f),
label.WidthAnchor.ConstraintEqualTo(label.IntrinsicContentSize.Width),
label.TopAnchor.ConstraintEqualTo(anchor:this.View.SafeAreaLayoutGuide.TopAnchor, constant:20.0f),
label.HeightAnchor.ConstraintEqualTo(label.IntrinsicContentSize.Height)
};
View.AddSubview(label);
}
}
异常堆栈跟踪:
at MvvmCross.Mvx.Resolve[TService] () [0x00006] in <8a077b300d9c484ab0471c2d21c3cb26>:0
at MvvmCross.Platforms.Ios.Views.MvxBindingViewControllerAdapter..ctor (MvvmCross.Platforms.Ios.Views.Base.IMvxEventSourceViewController eventSource) [0x00034] in <8a077b300d9c484ab0471c2d21c3cb26>:0
at MvvmCross.Platforms.Ios.Views.MvxViewControllerAdaptingExtensions.AdaptForBinding (MvvmCross.Platforms.Ios.Views.Base.IMvxEventSourceViewController view) [0x00007] in <8a077b300d9c484ab0471c2d21c3cb26>:0
at MvvmCross.Platforms.Ios.Views.MvxBaseTabBarViewController..ctor () [0x00006] in <8a077b300d9c484ab0471c2d21c3cb26>:0
at MvvmCross.Platforms.Ios.Views.MvxTabBarViewController..ctor () [0x00000] in <8a077b300d9c484ab0471c2d21c3cb26>:0
at MvvmCross.Platforms.Ios.Views.MvxTabBarViewController`1[TViewModel]..ctor () [0x00000] in <8a077b300d9c484ab0471c2d21c3cb26>:0
at PushNotifTest.iOS.Views.Main.MainViewController..ctor () <0x13d7f4970 + 0x0004a> in <1d11ff58113e46f6a5a9245eccb8c13f>:0
at PushNotifTest.iOS.AppDelegate.FinishedLaunching (UIKit.UIApplication application, Foundation.NSDictionary launchOptions) [0x00017] in /Users/dominik/Projekty/PushNotifTest/src/PushNotifTest.iOS/AppDelegate.cs:28
at (wrapper managed-to-native) UIKit.UIApplication.UIApplicationMain(int,string[],intptr,intptr)
at UIKit.UIApplication.Main (System.String[] args, System.IntPtr principal, System.IntPtr delegate) [0x00005] in /Library/Frameworks/Xamarin.iOS.framework/Versions/11.11.0.280/src/Xamarin.iOS/UIKit/UIApplication.cs:79
at UIKit.UIApplication.Main (System.String[] args, System.String principalClassName, System.String delegateClassName) [0x0002c] in /Library/Frameworks/Xamarin.iOS.framework/Versions/11.11.0.280/src/Xamarin.iOS/UIKit/UIApplication.cs:63
at PushNotifTest.iOS.Application.Main (System.String[] args) [0x00001] in /Users/dominik/Projekty/PushNotifTest/src/PushNotifTest.iOS/Main.cs:9
解决方案
你应该让 MvvmCross 为你做这件事。通常,您会指定应首先在IMvxAppStart
派生类中显示哪个 ViewModel。
public class AppStart : IMvxAppStart
{
private readonly IMvxNavigationService _navigationService;
public AppStart(IMvxNavigationService navigationService)
{
_navigationService = navigationService;
}
public void Start(object hint = null)
{
try
{
_navigationService.Navigate<MainViewModel>().GetAwaiter().GetResult();
}
catch (System.Exception e)
{
}
}
}
MainViewModel
与您的 ViewController 关联的 ViewModel 将在哪里,即MvxTabBarViewController
.
然后只需在 AppDelegate 中执行 MvvmCross 期望您执行的常规仪式:
public override bool FinishedLaunching(UIApplication application, NSDictionary launchOptions)
{
Window = new UIWindow(UIScreen.MainScreen.Bounds);
var setup = new Setup(this, Window);
setup.Initialize();
var startup = Mvx.Resolve<IMvxAppStart>();
startup.Start();
Window.MakeKeyAndVisible();
return true;
}
因此Setup
启动了 IoC 容器和其他相关的 MvvmCross 服务。然后IMvxAppStart
导航到第一个 ViewController。
根据您的属性MvxTabBarViewController
,它将被包装在 NavigationViewController 中。你可以用它来控制它MvxRootPresentationAttribute
。
[MvxRootPresentation(WrapInNavigationController = true)]
public partial class MainViewController : MvxTabBarViewController<MainViewModel>
编辑:
MvvmCross 6.0 没有做常规的仪式,而是让这部分变得更容易。所以删除 AppDelegate 的主体并像这样创建它:
[Register("AppDelegate")]
public partial class AppDelegate : MvxApplicationDelegate<MvxIosSetup<App>, App>
{
}
在您的核心中,您将拥有一个App
实现MvxApplication
. 这里调用RegisterAppStart<TViewModel>()
覆盖Initialize()
:
public class App : MvxApplication
{
public override void Initialize()
{
RegisterAppStart<RootViewModel>();
}
然后不要创建自己的AppStart
类。您可以Startup()
在App
课堂上进行 UI Bound 启动。
推荐阅读
- ios - UICollectionView 每个部分一行
- tensorflow - 为什么TensorFlow在调用1D卷积时要计算2D卷积?
- r - 带字符串标签的彩条指南
- python - 最小化和重新排列列表
- java - 我无法使用 Java Mail API 发送电子邮件 电子邮件已发送但未收到或未显示在已发送的电子邮件中
- python - 如何在 ubuntu 18 中使用 python-rsvg?
- pandas - 如何将左连接与两个单独的名称合并
- sql - SQL 中的货币转换
- sql-server - 如何在sql查询中解析多个命名空间xml
- python - Python If 语句 - 如果 mysql 的行值为 0,则忽略行