我正在尝试使用实时数据为视图模型编写单元测试。
登录视图模型.kt
class LoginViewModel @Inject constructor(
val context: Context
): ViewModel() {
val username = MutableLiveData<String>()
val password = MutableLiveData<String>()
val isLoginButtonEnabled = MediatorLiveData<Boolean>().apply {
fun combineLatest(): Boolean {
return !(username.value.isNullOrEmpty() || password.value.isNullOrEmpty())
}
addSource(username) { this.value = combineLatest() }
addSource(password) { this.value = combineLatest() }
}
init {
username.postValue("test")
password.postValue("test")
}
}
Run Code Online (Sandbox Code Playgroud)
登录ViewModelTest.kt
@RunWith(MockitoJUnitRunner::class)
class LoginViewModelTest {
@Rule
@JvmField
val instantTaskExecutorRole = InstantTaskExecutorRule()
private val context = mock(Context::class.java)
private val loginViewModel = LoginViewModel(context)
@Test
fun loginButtonDisabledOnEmptyUsername() {
val observer …Run Code Online (Sandbox Code Playgroud) android kotlin android-livedata android-viewmodel android-architecture-components
查看ViewModel文档,它说:
换句话说,这意味着如果ViewModel的所有者因配置更改(例如旋转)而被销毁,则不会销毁它.所有者的新实例将重新连接到现有的ViewModel.
如果引用它的活动被销毁,ViewModel如何不被破坏?一旦我们创建一个新活动,它是如何重新连接的?
java android android-viewmodel android-architecture-components
我正在学习kotlin和android架构组件.我在谷歌地图上有一个简单的地图切换按钮用例.
我想使用数据绑定将地图切换按钮标签绑定到我的ViewModel中的MutableLiveData字段.
我在Activity中的onCreate方法中设置MapViewModel中的mapType val.如果我理解正确,这应该触发mapLabel val由于使用Transformations.map而改变.
它不起作用......为什么?
这是我的版本:
MapViewModel.kt
class MapViewModel : ViewModel() {
val mapType: MutableLiveData<MapType> = MutableLiveData()
val mapLabel: LiveData<String> = Transformations.map(mapType, {
if (it == MapType.MAP) "SAT" else "MAP"
})
}
enum class MapType {
SAT, MAP
}
Run Code Online (Sandbox Code Playgroud)
activity_maps.xml
<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools">
<data>
<variable
name="vm"
type="uk.co.oliverdelange.wcr_android_kt.ui.map.MapViewModel" />
</data>
<FrameLayout
android:layout_width="match_parent"
android:layout_height="match_parent">
<fragment
android:id="@+id/map"
android:name="com.google.android.gms.maps.SupportMapFragment"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".MapsActivity">
<Button
android:id="@+id/map_toggle"
style="@style/Wcr_MapToggle"
android:layout_marginTop="110dp"
android:layout_marginEnd="12dp" …Run Code Online (Sandbox Code Playgroud) android kotlin android-livedata android-viewmodel android-architecture-components
我在我的android项目中使用匕首2(版本2.15)进行依赖项注入。除了注入viewModelFactory之外,其他一切都正常。我无法将其插入片段中。
@Inject
lateinit var viewModelFactory: ViewModelProvider.Factory
private val mainViewModel: StationsViewModel by lazy {
ViewModelProviders.of(this, viewModelFactory)
.get(StationsViewModel::class.java)
}
Run Code Online (Sandbox Code Playgroud)
我收到以下错误:
lateinit property viewModelFactory has not been initialized
Run Code Online (Sandbox Code Playgroud)
我有ViewModelModule:
@Module
abstract class ViewModelModule {
@Binds
@IntoMap
@ViewModelKey(MainViewModel::class)
internal abstract fun bindsMainViewModel(mainViewModel:
MainViewModel): ViewModel
@Binds
abstract fun bindViewModelFactory(factory: MyViewModelFactory):
ViewModelProvider.Factory
}
Run Code Online (Sandbox Code Playgroud)
ViewModelKey
@MustBeDocumented
@Target(
AnnotationTarget.FUNCTION,
AnnotationTarget.PROPERTY_GETTER,
AnnotationTarget.PROPERTY_SETTER
)
@Retention(AnnotationRetention.RUNTIME)
@MapKey
annotation class ViewModelKey(val value: KClass<out ViewModel>)
Run Code Online (Sandbox Code Playgroud)
和ViewModelFactory
@Singleton
class MyViewModelFactory @Inject constructor(
private val creators: Map<Class<out ViewModel>, @JvmSuppressWildcards Provider<ViewModel>>
) : ViewModelProvider.Factory {
override …Run Code Online (Sandbox Code Playgroud) android dependency-injection mvvm dagger-2 android-viewmodel
我是 Android Arch 组件的新手。我正在尝试使用 Android 视图模型和实时数据制作一个基本的待办事项应用程序。遵循 MVVM 模式时进行网络调用的最佳方法是什么?我需要progress bar在网络请求开始时显示 a并在调用完成时关闭它,如果出现错误,我需要显示snackbar带有相关错误消息的 a 。是否可以不使用AsyncTask?
远程存储库类:
public class RemoteRepository {
private APIService apiService;
public RemoteRepository (APIService apiService) {
this.apiService= apiService;
}
public LiveData<List<Project>> getList(String userId) {
final MutableLiveData<List<Project>> data = new MutableLiveData<>();
apiService.getList(userId).enqueue(new Callback<List<Project>>() {
@Override
public void onResponse(Call<List<Project>> call, Response<List<Project>> response) {
data.setValue(response.body());
}
@Override
public void onFailure(Call<List<Project>> call, Throwable t) {
// What to do to show snackbar in activity
}
});
return data;
} …Run Code Online (Sandbox Code Playgroud) android android-livedata android-viewmodel android-architecture-components
我基于发现使用Android体系结构组件和数据绑定的示例创建了代码。对我而言,这是一种新方法,并且对其进行编码的方式使得很难正确地使用单击的帖子的信息来打开新活动。
这是帖子的适配器
class PostListAdapter : RecyclerView.Adapter<PostListAdapter.ViewHolder>() {
private lateinit var posts: List<Post>
override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): PostListAdapter.ViewHolder {
val binding: ItemPostBinding = DataBindingUtil.inflate(
LayoutInflater.from(parent.context),
R.layout.item_post,
parent, false
)
return ViewHolder(binding)
}
override fun onBindViewHolder(holder: PostListAdapter.ViewHolder, position: Int) {
holder.bind(posts[position])
}
override fun getItemCount(): Int {
return if (::posts.isInitialized) posts.size else 0
}
fun updatePostList(posts: List<Post>) {
this.posts = posts
notifyDataSetChanged()
}
inner class ViewHolder(private val binding: ItemPostBinding) : RecyclerView.ViewHolder(binding.root) {
private val viewModel = PostViewModel()
fun bind(post: Post) …Run Code Online (Sandbox Code Playgroud) android mvvm android-recyclerview android-databinding android-viewmodel
我正在尝试在我目前正在编写的新应用程序中遵循 MVVM 模式。
基本上,它从我的 REST-Backend 获取 JSON 中的项目列表,并将其显示在我的片段内的 RecycleView 中。
我创建了一个存储库,它获取数据并将其移交给具有由片段观察到的 LiveData 的 ViewModel。
这一切正常。
但是:每个项目还有一个图标的 url。获取列表后,对于每个项目,我都想将此 url 中的图标加载到 ImageView 中。
实际上,我使用 Glide 直接(异步)将图标加载到相应的 ImageView 中 - 这对 UX 和性能都有好处(在我看来),因为用户在图标在后台加载时已经看到了数据
我的问题:
直接在片段中使用 Glide 会破坏 MVVM 模式吗?
什么是替代方法?
例如在存储库中加载图标,每次获取图标时更新 RecycleView(性能不佳)?
android mvvm android-glide android-viewmodel android-architecture-components
我浏览了许多与带有数据绑定的 MVVM 模型相关的博客。由于与 ViewModel 的数据绑定使得编写 junit 测试用例变得容易。
我想知道,我怎么能实现监听器事件一样OnTouchListener,OnClickListener,OnFocusChangeListener在这将使容易编写单元测试用例视图模型数据绑定。
我已经使用黄油刀库进行绑定,并通过它执行OnTouch事件,我的问题是,这是在 Activity 中实现侦听器而不是直接在 ViewModel 中实现它的正确方法吗?
请参考以下代码以了解具有 MVVM 结构的 LoginScreen:
登录ActivityNew.java
public class LoginActivityNew extends AppCompatActivity {
@BindView(R.id.et_password)
AppCompatEditText etPassword;
private LoginViewModel loginViewModel;
ActivityLoginBinding binding;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
binding = DataBindingUtil.setContentView(this, R.layout.activity_login);
loginViewModel = ViewModelProviders.of(this).get(LoginViewModel.class);
binding.setViewModel(loginViewModel);
binding.setLifecycleOwner(this);
ButterKnife.bind(this);
binding.buttonLogin.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Common common = new Common(getApplicationContext());
common.isInternetAvailable(LoginActivityNew.this, new Common.InternetStateListener() {
@Override
public …Run Code Online (Sandbox Code Playgroud) android mvvm android-databinding android-mvvm android-viewmodel
我创建了一个抽象BaseFragment类,它将被其他具体Fragment类扩展。我想注入ViewModel我BaseFragment使用的Koin. 这是我的 BaseFragment:
abstract class BaseFragment<out VM : BaseViewModel, DB : ViewDataBinding>(private val mViewModelClass: Class<VM>) : Fragment() {
val viewModel: VM by viewModel()
open lateinit var binding: DB
fun init(inflater: LayoutInflater, container: ViewGroup) {
binding = DataBindingUtil.inflate(inflater, getLayoutRes(), container, false)
}
open fun init() {}
@LayoutRes
abstract fun getLayoutRes(): Int
override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?,
savedInstanceState: Bundle?): View {
init(inflater, container!!)
init()
super.onCreateView(inflater, container, savedInstanceState)
return binding.root
}
open …Run Code Online (Sandbox Code Playgroud) 当有人开始在 EditText 中书写时,我试图启用该按钮。
我将按钮的启用属性设置为android:enabled ="@{editText.text.length() > 0}"
这工作得很好。但是当我尝试使用两种方式数据绑定将 EditText 绑定到 ViewModel 时它停止工作。
正是这条线产生了差异。
android:text="@={signUpViewModel.Phone}"
Run Code Online (Sandbox Code Playgroud)
没有这个,它工作正常!可能是什么原因?
下面是完整的 XML。
<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools">
<data>
<variable
name="signUpViewModel"
type="com.qenetech.africapuzzle.signup.SignUpViewModel" />
<variable
name="editText_verify_phone_number"
type="android.widget.EditText" />
</data>
<android.support.constraint.ConstraintLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
tools:context=".signup.verify_phone.EnterPhoneFragment">
<Spinner
android:id="@+id/spinner_country"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_marginStart="16dp"
android:layout_marginTop="60dp"
android:layout_marginEnd="16dp"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
<LinearLayout
android:id="@+id/linearLayout"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_marginTop="60dp"
android:gravity="center"
android:orientation="horizontal"
android:weightSum="10"
app:layout_constraintEnd_toEndOf="@+id/spinner_country"
app:layout_constraintHorizontal_bias="0.0"
app:layout_constraintStart_toStartOf="@+id/spinner_country"
app:layout_constraintTop_toBottomOf="@+id/spinner_country">
<android.support.design.widget.TextInputLayout
android:layout_width="0dp"
android:layout_weight="2"
android:layout_height="wrap_content"
android:textColorHint="@android:color/darker_gray">
<android.support.design.widget.TextInputEditText
android:id="@+id/editText_verify_phone_country_code"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:ems="10"
android:text="@={signUpViewModel.CountryCode}"
android:inputType="phone" />
</android.support.design.widget.TextInputLayout>
<android.support.design.widget.TextInputLayout
android:id="@+id/etPhoneLayout"
android:layout_width="0dp"
android:layout_weight="8" …Run Code Online (Sandbox Code Playgroud) data-binding android mvvm android-databinding android-viewmodel
android ×10
android-architecture-components ×5
mvvm ×5
kotlin ×3
android-mvvm ×1
dagger-2 ×1
data-binding ×1
java ×1
koin ×1