Это очень просто благодаря @ Эрику Брауну и литералам функций Котлина с приемником
Создать SingleLiveEvent
Создать LiveMessageEvent
class LiveMessageEvent<T> : SingleLiveEvent<(T.() -> Unit)?>() {
fun setEventReceiver(owner: LifecycleOwner, receiver: T) {
observe(owner, Observer { event ->
if ( event != null ) {
receiver.event()
}
})
}
fun sendEvent(event: (T.() -> Unit)?) {
value = event
}
}
Создать интерфейс
interface ActivityNavigation {
fun startActivityForResult(intent: Intent?, requestCode: Int)
}
** Теперь пришло время реализовать! **
В вашем LoginViewModel
const val GOOGLE_SIGN_IN : Int = 9001
class LoginViewModel @Inject constructor(
private val loginRepository: LoginRepository,
private val googleSignInClient: GoogleSignInClient
): ViewModel() {
val startActivityForResultEvent = LiveMessageEvent<ActivityNavigation>()
..
//Called on google login button click
fun googleSignUp() {
val signInIntent = googleSignInClient.signInIntent
startActivityForResultEvent.sendEvent { startActivityForResult(signInIntent, GOOGLE_SIGN_IN) }
}
//Called from Activity receving result
fun onResultFromActivity(requestCode: Int, resultCode: Int, data: Intent?) {
when(requestCode) {
GOOGLE_SIGN_IN -> {
val task = GoogleSignIn.getSignedInAccountFromIntent(data)
googleSignInComplete(task)
}
..
}
}
private fun googleSignInComplete(completedTask: Task<GoogleSignInAccount>) {
try {
val account = completedTask.getResult(ApiException::class.java)
account?.apply {
// .. Store user details
emitUiState(
showSuccess = Event(R.string.login_successful)
)
}
}catch (e: ApiException) {
emitUiState(
showError = Event(R.string.login_failed)
)
}
}
В вашем логинеActivty
//Called from onCreate once the ViewModel is initialized.
private fun subscribeUi() {
//this sets the LifeCycler owner and receiver
viewModel.startActivityForResultEvent.setEventReceiver(this, this)
..
}
public override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
viewModel.onResultFromActivity(requestCode,resultCode,data)
super.onActivityResult(requestCode, resultCode, data)
}
# При таком подходе LoginViewModel действует как связующее звено между представлением (LoginActivity) и моделью (GoogleSignInClient), а представление отвечает только за отображение событий пользовательского интерфейса.