这是我使用 MVVM(+数据绑定)和 Dagger-2.11-rc2 创建Adapter
的方式:
适配器:
public class ItemAdapter extends RecyclerView.Adapter<BindableViewHolder<ViewDataBinding>>{
private static int TYPE_A = 0;
private static int TYPE_B = 1;
...
@Override
public BindableViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
if (viewType == TYPE_A) {
return new ItemViewHolder(new ItemRowView(parent.getContext()).getBinding());
}
...
}
@Override
public void onBindViewHolder(BindableViewHolder holder, int position) {
if (holder.getItemViewType() == TYPE_A) {
((ItemViewHolderBinding) holder.getBinding()).getViewModel().setItemModel(((ItemModel) getItem(position)));
}
...
}
private static class ItemViewHolder extends BindableViewHolder<ItemViewHolderBinding> {
ItemViewHolder(ItemViewHolderBinding binding) {
super(binding);
}
}
}
BindableViewHolder:
public abstract class BindableViewHolder<ViewBinding extends ViewDataBinding> extends RecyclerView.ViewHolder {
private ViewBinding mBinding;
public BindableViewHolder(ViewBinding binding) {
super(binding.getRoot());
mBinding = binding;
}
public ViewBinding getBinding(){
return mBinding;
}
}
由于我使用Dagger,我不会在ViewModels
内创建Adapter
,而是在各自的Android.View
内创建(注入)它们。我认为这是有道理的,因为我的Adapter
可能有X Android.View
类型,这些视图可以有Y ViewModel
等等。
基本视角:
public abstract class BaseView<ViewBinding extends ViewDataBinding, ViewModel extends BaseViewModel> extends FrameLayout {
@Inject
ViewModel mViewModel;
protected ViewBinding mBinding;
protected abstract void initBinding(final ViewBinding binding, final ViewModel viewModel);
...
private void initView(Context context) {
ViewInjection.inject(this);
mBinding = DataBindingUtil...
initBinding(mBinding, mViewModel);
...
}
...
}
BaseViewModel :
public class BaseViewModel extends BaseObservable {...}
ItemRowView(或任何视图):
public class ItemRowView extends BaseView<ItemRowViewBinding, ItemRowViewModel> {
@Inject
ViewModelA mViewModelA;
@Inject
ViewModelB mViewModelB;
...
@Override
protected void initBinding(ItemRowViewBinding binding, ItemRowViewModel viewModel) {
binding.setViewModel(viewModel);
binding.setViewModelA(mViewModelA);
binding.setViewModelB(mViewModelB);
...
}
}
现在,这种方法适用于Activities,Fragments等,但是当我使用Views时,我必须创建一个 ViewInjecton ,因为Dagger并没有开箱即用。 This is how I do it(阅读直至您已经&#34; ViewInjection几乎是其他注射者的副本。&#34; )
我的问题是(这是):这是一个好方法吗?我正确使用MVVM和Dagger吗?有没有更好的方法来实现这一点而不创建 ViewInjecton (和使用Dagger-2.11)?
感谢您的时间。
ps:我已经使用了Adapter
示例,但如果我想使用Views而不是Fragments,这种方法是一样的。使用Adapters
,您只能使用观看次数。
答案 0 :(得分:2)
已经讨论过是否应该在this question内注入视图内部。
由于我正在使用Dagger,我不会在适配器内部创建ViewModel,而是在各自的Android.View中创建(注入)它们。我想这很有意义,因为我的Adapter可能有X个Android.View类型,那些视图可以有Y ViewModel等等......
我个人认为这有点问题,如果我正在使用该代码的团队工作,我希望层之间更大程度的分离。至少,
Adapter
可以直接处理模型图层,如果它与“项目”图层很容易相关,即List
的支持内容RecyclerView
。 ViewModel
的{{1}}应该非常轻巧,不需要注射。它应该基本上是一包属性,可以轻松转换为视图的某些属性(例如,RecyclerView.ViewHolder
,setText()
),并且可以获取/设置。这些可以使用适配器中setColor()
方法内的new
关键字创建。如果这很困难,您可以提取Factory(onBindViewHolder
)并将其作为Adapter的依赖项注入。 简而言之,模型数据对象应该是“哑”。个人ViewModelFactory
的{{1}}也是如此。 ViewModel
可以是“智能”的,并且可以采用经过测试的“智能”依赖项(如果需要,可以使用ViewHolder
),并且可以使用Dagger 2将此Adapter
注入到您的Activity或Fragment中。
答案 1 :(得分:0)
虽然我同意 David's answer 不应该这样做,但如果您仍然想这样做,可以通过参加活动:
override val activity: FragmentActivity by lazy {
try {
context as FragmentActivity
} catch (exception: ClassCastException) {
throw ClassCastException("Please ensure that the provided Context is a valid FragmentActivity")
}
}
override var viewModel = ViewModelProviders.of(activity).get(SharedViewModel::class.java)
对此进行了更详细的讨论 here。