3

我有 2 ViewModels(ConfigurationViewModelEditConfigurationViewModel)。在ConfigurationViewModel我有以下代码:

    public ConfigurationViewModel()
    {
        NewConfigCommand = new MvxRelayCommand(DoNewConfig);
        EditConfigCommand = new MvxRelayCommand<ConfigurationSet>(DoEditConfig);
    }

    private void DoNewConfig()
    {
        this.RequestNavigate<EditConfigurationViewModel>();
    }

    private void DoEditConfig(ConfigurationSet config)
    {
        this.RequestNavigate<EditConfigurationViewModel>(new { id = config.Id.ToString() });
    }

在 EditConfigurationViewModel 我有以下代码:

    public EditConfigurationViewModel()
    {
        Configuration = new ConfigurationSet();
    }

    public EditConfigurationViewModel(string id)
    {
        Configuration = ConfigDataStore.GetConfiguration(Guid.Parse(id));
    }

我想要实现的是非常简单的......在触发ConfigurationViewModelNewConfigCommand,我想导航到EditConfigurationViewModel,并使用无参数构造函数。当EditConfigCommand被触发时,我想使用接收string.

这段代码的问题是,无论触发什么命令,总是使用无参数构造函数,并且代码永远不会到达另一个构造函数。

我做了一些实验,通过删除无参数构造函数,结果是调用了另一个构造函数,我得到了预期的结果EditConfigurationCommand,但是如果我尝试触发NewConfigurationCommand异常,则由于无参数构造函数的不存在(到目前为止,一切都很好)。

不幸的是,此时我没有安装 VS2010,所以我无法通过 PCL 代码进行调试......我做了一些“眼睛调试”并找到了这个类MvxViewModelLocator。我认为问题出在此处。也许在DoLoad尝试获取MethodInfo...的方法中

在这一点上,我只想知道我是否做错了什么,或者这是否是预期的结果。同时我想我会冒险安装VS2010并祈祷它不会破坏任何东西......

4

1 回答 1

4

在 PCL 调试问题上,为什么不直接添加 Win8 或 WP7/8 UI - 然后您可以通过 PCL 代码进行调试...


关于主要问题 - 关于如何使用多个构造函数......我建议你不要

对我来说,edit 和 new 是两个不同的视图和两个不同的视图模型——它们可能共享共同的属性和共同的布局——但这可以通过继承、使用 UserControls、使用includeaxml 等来实现。

有关我通常用于新建和编辑的示例,请参阅https://github.com/slodge/MvvmCross/tree/vnext/Sample%20-%20CustomerManagement/CustomerManagement/CustomerManagement/ViewModels


如果您确实坚持使用一个视图模型,那么您可以考虑为 New 使用“神奇值” - 例如,如果 Guid.Empty 被传递,那么这意味着新的?


或者,您可以删除无参数构造函数,并为第二个构造函数添加默认值:

public EditConfigurationViewModel(string id = null)
{
    Guid value;
    if (id == null || !Guid.TryParse(id, out value))
    {
        Configuration = new ConfigurationSet();
    }
    else
    {
        Configuration = ConfigDataStore.GetConfiguration(value);
    }
}

我认为这会奏效吗?


最后,如果这些都不适合您,那么您可以考虑覆盖 ViewModel 构造机制。

为了帮助解决这个问题,最近有一篇相当详细的文章介绍了如何为 MvvmCross 编写自己的默认 ViewModelLocator - 请参阅http://slodge.blogspot.co.uk/2013/01/navigating-between-viewmodels-by-more.html

使用这种方法,您可以创建一个更加自定义的导航模型 - 或者如果这是唯一的特殊视图模型,那么我怀疑您可以创建一个默认的 viewModelLocator,例如:

public class MyViewModelLocator
    : MvxDefaultViewModelLocator
{
    public override bool TryLoad(Type viewModelType, IDictionary<string, string> parameterValueLookup,
                                 out IMvxViewModel model)
    {
        if (viewModelType == typeof(EditConfigurationViewModel))
        {
            string id;
            if (parameterValueLookup.TryGetValue("id", out id))
            {
                model = new EditConfigurationViewModel(id);
            }
            else
            {
                model = new EditConfigurationViewModel();
            }
            return true;
        }
        return base.TryLoad(viewModelType, parameterValueLookup, IMvxViewModel model);
    }
}

并使用以下方法在 App.cs 中注册该定位器:

protected override IMvxViewModelLocator CreateDefaultViewModelLocator()
{
     return new MyViewModelLocator();
}
于 2013-01-16T21:15:01.343 回答