我的 Model
是一个包含(例如)Value
的通用类可以是 int、float、string、bool 等的属性。所以自然地,此类表示为 Model<T>
。 .为了收藏Model<T>
实现接口(interface) IModel
, 尽管 IModel
本身没有任何内容。
我的 ViewModel
包含 Model<T>
的实例并通过ViewModel
传入的构造函数。我还想知道什么T
在 ViewModel 中,所以当我公开 Model
时到 View
我知道 Model
的数据类型埋了Value
属性(property)。 ViewModel 的类最终如下所示:
class ViewModel<T>
{
private Model<T> _model;
public ViewModel(Model<T> model) { ....blah.... }
public T ModelsValue {get; set; }
}
这工作正常,但有限制。所以现在我需要公开 IModels
的集合随着变化 Ts
到我的View
,所以我正在尝试设置一个 ObservableCollection
的 <strong>new</strong> ViewModel<T>s
更改列表 IModels
.问题是,我不知道如何获得 T
来自 Model<T>
来自 IModel
build ViewModel<T>(Model<T>)
在运行时。
在 VS2010 调试器中,我可以将鼠标悬停在任何 IModel
上对象并查看其完整 Model<int>
例如在运行时,所以我知道数据在那里。
有什么想法吗?
最佳答案
这是我用于 View 模型集合的内容:
前言:
您的 View 模型对象可以是弱类型的。给IModel
一处特性object Value {get;}
并在 ModelViewModel : ViewModel<IModel>
中公开它你用于所有 IModel
对象(参见下面我的 ViewModel<T>
实现)。如果你有 ObservableCollection<IModel>
的各种组合, ICollection<Model<T>>
等,此处显示的框架是救星。如果您仍然需要通用 View 模型,您可以派生一个 ModelViewModel<T> : ModelViewModel
这需要 Model<T>
在它的构造函数中。创建适当类型的逻辑将进入传递给 ViewModelCollection.Create
的转换器中以下。 请注意,此设计会带来性能损失。
ModelViewModel CreateModelViewModel(IModel model)
{
Type viewModelType = typeof(ModelViewModel<>).MakeGenericType(model.Type);
ModelViewModel viewModel = Activator.CreateInstance(viewModelType, model);
return viewModel;
}
示例用法:
public class CatalogViewModel : ViewModel<ICatalog>
{
public CatalogViewModel(ICatalog catalog)
: base(catalog)
{
Func<ICatalogProduct, ProductViewModel> viewModelFactory = CreateProductViewModel;
this.Products = ViewModelCollection.Create(catalog.Products, viewModelFactory);
}
public ICollection<ProductViewModel> Products
{
get;
private set;
}
private ProductViewModel CreateProductViewModel(ICatalogProduct product)
{
return new ProductViewModel(product, this);
}
}
好处:
- 使用延迟实现以允许在树中进行高效甚至递归绑定(bind)。
- View 模型集合仅实现
INotifyCollectionChanged
如果基础模型集合实现INotifyCollectionChanged
.
类概述(链接到 github 的完整实现):
ViewModel<TModel>
:我的 View 模型类的基类。公开一个Model
我在 View 模型的支持代码中使用的属性。ObservableViewModelCollection<TViewModel, TModel>
:惰性(实际上目前不是,但绝对应该是),从模型到 View 模型的可观察映射。工具INotifyCollectionChanged
.ViewModelCollection<TViewModel, TModel>
: Lazy 从 TModel 集合到TViewModel
集合的映射.ViewModelCollection
: 静态助手 - 返回ICollection<TViewModel>
, 使用ObservableViewModelCollection<TViewModel, TModel>
当源集合实现INotifyCollectionChanged
时, 否则使用ViewModelCollection<TViewModel, TModel>
.
一些可能对您的 View 模型集合有用的额外类型:
ConcatCollection
:与 ViewModelCollection 一样,这包括一个静态帮助程序来自动选择适当的实现。 ConcatCollection 通过直接绑定(bind)到源集合来连接集合。
这是我如何使用此类型公开 Children
的示例将属性添加到 View ,同时一直维护我的可观察集合,直到返回原始来源。
public class ProductViewModel : ViewModel<IProduct>
{
public ProductViewModel(IProduct product)
: base(product)
{
Func<IProduct, ProductViewModel> productViewModelFactory = CreateProductViewModel;
Func<IRelease, ReleaseViewModel> releaseViewModelFactory = CreateReleaseViewModel;
this.Products = ViewModelCollection.Create(product.Products, productViewModelFactory);
this.Releases = ViewModelCollection.Create(product.Releases, releaseViewModelFactory);
this.Children = ConcatCollection.Create<object>((ICollection)this.Products, (ICollection)this.Releases);
}
public IList<ProductViewModel> Products
{
get;
private set;
}
public IList<ReleaseViewModel> Releases
{
get;
private set;
}
public IEnumerable<object> Children
{
get;
private set;
}
private ProductViewModel CreateProductViewModel(IProduct product)
{
return new ProductViewModel(product);
}
private ReleaseViewModel CreateReleaseViewModel(IRelease release)
{
return new ReleaseViewModel(release);
}
}
关于带有 MVVM 的 C# 泛型,将 T 从 <T> 中拉出,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2698910/