1

我正在学习 MVVM 模式,现在有一个问题。我有一个带有 imageview 的活动 A,还有一个带有相同 imageview 但在另一个地方更大的活动 B。当在活动 A 中单击 imageview 时,我想启动活动 B,并且 imageview 应该是共享元素,以通过漂亮的动画实现我想要的。

使用数据绑定我处理在我的视图模型中单击图像视图:

<de.hdodenhof.circleimageview.CircleImageView
                    android:layout_width="60dp"
                    android:layout_height="60dp"
                    android:onClick="@{(v) -> user.onAvatarClick(v)}"
                    android:scaleType="centerCrop"
                    android:src="@{user.photoUrl}"
                    android:transitionName="@string/avatar_transition" />

在我的视图模型中,我应该写这样的东西:

public void onAvatarClick(View view) {
    Intent intent = new Intent(context, AvatarActivity.class);
    ActivityOptionsCompat options = ActivityOptionsCompat.makeSceneTransitionAnimation(activity, view, view.getTransitionName());
    context.startActivity(intent, options.toBundle());
}

但是,问题是我的视图模型对活动一无所知。而且我不知道如何做我想做的事。

一种可能的解决方案是使用 startActivityB(View view) 之类的方法创建接口,并在活动中实现它并将其设置为字段,例如 viewModelListener。然后我可以写类似的东西:

public void onAvatarClick(View view) {
    viewModelListener.startActivityB(view);
}

但在这种情况下,我的视图模型将引用视图,它打破了 MVVM 的主要思想,对吧?

那么,使用 MVVM 架构通过共享元素开始新活动的正确方法是什么?

4

1 回答 1

1

这可以通过 Android 中的实时数据来完成。您可以观察实时数据的变化并相应地导航到其他活动。

这里的例子

在您单击按钮的 xml 文件中

android:onClick="@{()->homeActivityViewModel.openNewActivity()}"

在您的 viewModel 类中

private final MutableLiveData<Boolean> openNewScreen = new MutableLiveData<>();

 //function that is binded to xml
public void openNewActivity() {
    openNewScreen.setValue(true);
}

public MutableLiveData<Boolean> getNewScreen() {
    return openNewScreen;
}

在你的活动中

 homeActivityViewModel.getNewScreen().observe(this,
            start -> {
                if (start) {
                    Intent intent = new Intent(this, NewActivity.class);
                    startActivity(intent);
                } });
于 2018-08-28T18:49:32.843 回答