ایجاد ViewModels با وابستگی بخشی از Android Jetpack .
با پیروی از بهترین روشهای تزریق وابستگی ، ViewModels میتواند وابستگیها را به عنوان پارامتر در سازنده خود در نظر بگیرد. اینها بیشتر از انواع لایه های دامنه یا داده هستند. از آنجایی که فریم ورک ViewModels را فراهم می کند، مکانیسم خاصی برای ایجاد نمونه هایی از آنها لازم است. آن مکانیسم رابط ViewModelProvider.Factory
است. فقط پیادهسازیهای این رابط میتوانند ViewModels را در محدوده مناسب نمونهسازی کنند .
ViewModels با CreationExtras
اگر یک کلاس ViewModel وابستگی هایی را در سازنده خود دریافت می کند، کارخانه ای ارائه کنید که رابط ViewModelProvider.Factory
را پیاده سازی کند. برای ارائه یک نمونه جدید از ViewModel، تابع create(Class<T>, CreationExtras)
را لغو کنید.
CreationExtras
به شما امکان می دهد به اطلاعات مربوطه دسترسی داشته باشید که به نمونه سازی ViewModel کمک می کند. در اینجا لیستی از کلیدهایی است که می توان از طریق موارد اضافی به آنها دسترسی داشت:
کلید | کارکرد |
---|---|
ViewModelProvider.NewInstanceFactory.VIEW_MODEL_KEY | دسترسی به کلید سفارشی که به ViewModelProvider.get() ارسال کرده اید را فراهم می کند. |
ViewModelProvider.AndroidViewModelFactory.APPLICATION_KEY | دسترسی به نمونه کلاس Application را فراهم می کند. |
SavedStateHandleSupport.DEFAULT_ARGS_KEY | دسترسی به بستهای از آرگومانهایی را که باید برای ساخت SavedStateHandle استفاده کنید، فراهم میکند. |
SavedStateHandleSupport.SAVED_STATE_REGISTRY_OWNER_KEY | دسترسی به SavedStateRegistryOwner که برای ساخت ViewModel استفاده می شود را فراهم می کند. |
SavedStateHandleSupport.VIEW_MODEL_STORE_OWNER_KEY | دسترسی به ViewModelStoreOwner که برای ساخت ViewModel استفاده می شود را فراهم می کند. |
برای ایجاد یک نمونه جدید از SavedStateHandle
، از تابع CreationExtras.createSavedStateHandle()
.createSavedStateHandle()) استفاده کنید و آن را به ViewModel ارسال کنید.
در زیر نمونهای از نحوه ارائه نمونهای از ViewModel است که یک مخزن را به کلاس Application
و SavedStateHandle
به عنوان وابستگی میبرد:
کاتلین
import androidx.lifecycle.SavedStateHandle
import androidx.lifecycle.ViewModel
import androidx.lifecycle.ViewModelProvider
import androidx.lifecycle.ViewModelProvider.AndroidViewModelFactory.Companion.APPLICATION_KEY
import androidx.lifecycle.createSavedStateHandle
import androidx.lifecycle.viewmodel.CreationExtras
class MyViewModel(
private val myRepository: MyRepository,
private val savedStateHandle: SavedStateHandle
) : ViewModel() {
// ViewModel logic
// ...
// Define ViewModel factory in a companion object
companion object {
val Factory: ViewModelProvider.Factory = object : ViewModelProvider.Factory {
@Suppress("UNCHECKED_CAST")
override fun <T : ViewModel> create(
modelClass: Class<T>,
extras: CreationExtras
): T {
// Get the Application object from extras
val application = checkNotNull(extras[APPLICATION_KEY])
// Create a SavedStateHandle for this ViewModel from extras
val savedStateHandle = extras.createSavedStateHandle()
return MyViewModel(
(application as MyApplication).myRepository,
savedStateHandle
) as T
}
}
}
}
جاوا
import static androidx.lifecycle.SavedStateHandleSupport.createSavedStateHandle;
import static androidx.lifecycle.ViewModelProvider.AndroidViewModelFactory.APPLICATION_KEY;
import androidx.lifecycle.SavedStateHandle;
import androidx.lifecycle.ViewModel;
import androidx.lifecycle.viewmodel.ViewModelInitializer;
public class MyViewModel extends ViewModel {
public MyViewModel(
MyRepository myRepository,
SavedStateHandle savedStateHandle
) { /* Init ViewModel here */ }
static final ViewModelInitializer<MyViewModel> initializer = new ViewModelInitializer<>(
MyViewModel.class,
creationExtras -> {
MyApplication app = (MyApplication) creationExtras.get(APPLICATION_KEY);
assert app != null;
SavedStateHandle savedStateHandle = createSavedStateHandle(creationExtras);
return new MyViewModel(app.getMyRepository(), savedStateHandle);
}
);
}
سپس، هنگام بازیابی نمونه ای از ViewModel، می توانید از این کارخانه استفاده کنید:
کاتلین
import androidx.activity.viewModels
class MyActivity : AppCompatActivity() {
private val viewModel: MyViewModel by viewModels { MyViewModel.Factory }
// Rest of Activity code
}
جاوا
import androidx.appcompat.app.AppCompatActivity;
import androidx.lifecycle.ViewModelProvider;
public class MyActivity extends AppCompatActivity {
MyViewModel myViewModel = new ViewModelProvider(
this,
ViewModelProvider.Factory.from(MyViewModel.initializer)
).get(MyViewModel.class);
// Rest of Activity code
}
Jetpack Compose
import androidx.lifecycle.viewmodel.compose.viewModel
@Composable
fun MyScreen(
modifier: Modifier = Modifier,
viewModel: MyViewModel = viewModel(factory = MyViewModel.Factory)
) {
// ...
}
روش دیگر، از ViewModel factory DSL برای ایجاد کارخانه با استفاده از یک API اصطلاحی Kotlin استفاده کنید:
import androidx.lifecycle.SavedStateHandle
import androidx.lifecycle.ViewModel
import androidx.lifecycle.ViewModelProvider
import androidx.lifecycle.ViewModelProvider.AndroidViewModelFactory.Companion.APPLICATION_KEY
import androidx.lifecycle.createSavedStateHandle
import androidx.lifecycle.viewmodel.initializer
import androidx.lifecycle.viewmodel.viewModelFactory
class MyViewModel(
private val myRepository: MyRepository,
private val savedStateHandle: SavedStateHandle
) : ViewModel() {
// ViewModel logic
// Define ViewModel factory in a companion object
companion object {
val Factory: ViewModelProvider.Factory = viewModelFactory {
initializer {
val savedStateHandle = createSavedStateHandle()
val myRepository = (this[APPLICATION_KEY] as MyApplication).myRepository
MyViewModel(
myRepository = myRepository,
savedStateHandle = savedStateHandle
)
}
}
}
}
Factories برای ViewModel نسخه قدیمی تر از 2.5.0
اگر از نسخهای از ViewModel قدیمیتر از 2.5.0 استفاده میکنید، باید کارخانههایی را از زیرمجموعهای از کلاسها تهیه کنید که ViewModelProvider.Factory
گسترش داده و تابع create(Class<T>)
را پیادهسازی کند. بسته به اینکه ViewModel به چه وابستگی هایی نیاز دارد، یک کلاس متفاوت باید از:
- در صورت نیاز به کلاس
Application
AndroidViewModelFactory
. -
AbstractSavedStateViewModelFactory
اگرSavedStateHandle
باید به عنوان یک وابستگی ارسال شود.
اگر Application
یا SavedStateHandle
نیاز نیست، به سادگی از ViewModelProvider.Factory
گسترش دهید.
مثال زیر از AbstractSavedStateViewModelFactory
برای ViewModel استفاده می کند که یک مخزن و یک نوع SavedStateHandle
را به عنوان وابستگی می گیرد:
کاتلین
class MyViewModel(
private val myRepository: MyRepository,
private val savedStateHandle: SavedStateHandle
) : ViewModel() {
// ViewModel logic ...
// Define ViewModel factory in a companion object
companion object {
fun provideFactory(
myRepository: MyRepository,
owner: SavedStateRegistryOwner,
defaultArgs: Bundle? = null,
): AbstractSavedStateViewModelFactory =
object : AbstractSavedStateViewModelFactory(owner, defaultArgs) {
@Suppress("UNCHECKED_CAST")
override fun <T : ViewModel> create(
key: String,
modelClass: Class<T>,
handle: SavedStateHandle
): T {
return MyViewModel(myRepository, handle) as T
}
}
}
}
جاوا
import androidx.annotation.NonNull;
import androidx.lifecycle.AbstractSavedStateViewModelFactory;
import androidx.lifecycle.SavedStateHandle;
import androidx.lifecycle.ViewModel;
public class MyViewModel extends ViewModel {
public MyViewModel(
MyRepository myRepository,
SavedStateHandle savedStateHandle
) { /* Init ViewModel here */ }
}
public class MyViewModelFactory extends AbstractSavedStateViewModelFactory {
private final MyRepository myRepository;
public MyViewModelFactory(
MyRepository myRepository
) {
this.myRepository = myRepository;
}
@SuppressWarnings("unchecked")
@NonNull
@Override
protected <T extends ViewModel> T create(
@NonNull String key, @NonNull Class<T> modelClass, @NonNull SavedStateHandle handle
) {
return (T) new MyViewModel(myRepository, handle);
}
}
سپس، می توانید از کارخانه برای بازیابی ViewModel خود استفاده کنید:
کاتلین
import androidx.activity.viewModels
class MyActivity : AppCompatActivity() {
private val viewModel: MyViewModel by viewModels {
MyViewModel.provideFactory((application as MyApplication).myRepository, this)
}
// Rest of Activity code
}
جاوا
import androidx.appcompat.app.AppCompatActivity;
import androidx.lifecycle.ViewModelProvider;
public class MyActivity extends AppCompatActivity {
MyViewModel myViewModel = new ViewModelProvider(
this,
ViewModelProvider.Factory.from(MyViewModel.initializer)
).get(MyViewModel.class);
// Rest of Activity code
}
Jetpack Compose
import androidx.lifecycle.viewmodel.compose.viewModel
@Composable
fun MyScreen(
modifier: Modifier = Modifier,
viewModel: MyViewModel = viewModel(
factory = MyViewModel.provideFactory(
(LocalContext.current.applicationContext as MyApplication).myRepository,
owner = LocalSavedStateRegistryOwner.current
)
)
) {
// ...
}
برای شما توصیه می شود
- توجه: وقتی جاوا اسکریپت خاموش است، متن پیوند نمایش داده می شود
- ماژول حالت ذخیره شده برای ViewModel
- حالت های رابط کاربری را ذخیره کنید
- نمای کلی LiveData