From b05ecc9d8fac79c0f6f3625572e31899d9072002 Mon Sep 17 00:00:00 2001 From: winter223 Date: Sat, 30 Apr 2022 13:06:31 +0900 Subject: [PATCH] =?UTF-8?q?[#30]=20=EB=A1=9C=EA=B7=B8=EC=9D=B8=20=EA=B8=B0?= =?UTF-8?q?=EB=8A=A5=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit - 신규 유저일 경우, IdToken, ProviderType 을 ProfileEditActivity로 전달 - ProfileEditActivity 에서 다음 버튼을 눌렀을 때 IdToken 과 ProviderType, 입력한 프로필 정보를 바탕으로 회원가입 진행할 수 있도록 밑작업 --- app/src/main/AndroidManifest.xml | 5 ++- .../moyeorun_android/login/ProviderType.kt | 9 ++++ .../login/data/LoginRepository.kt | 12 ++++-- .../login/data/model/LoginRequest.kt | 6 +++ .../login/data/{ => model}/LoginResponse.kt | 2 +- .../login/data/{ => model}/Token.kt | 2 +- .../login/ui/LoginActivity.kt | 27 ++++++------ .../moyeorun_android/login/ui/LoginEvent.kt | 8 +++- .../login/ui/LoginViewModel.kt | 15 ++++--- .../profile/ProfileEditActivity.kt | 41 +++++++++++++++++-- .../profile/ProfileEditViewModel.kt | 36 +++++++++++++--- .../profile/ProfileUiModel.kt | 11 ++++- .../profile/SignUpMetaData.kt | 16 ++++++++ 13 files changed, 151 insertions(+), 39 deletions(-) create mode 100644 app/src/main/java/com/moyerun/moyeorun_android/login/ProviderType.kt create mode 100644 app/src/main/java/com/moyerun/moyeorun_android/login/data/model/LoginRequest.kt rename app/src/main/java/com/moyerun/moyeorun_android/login/data/{ => model}/LoginResponse.kt (77%) rename app/src/main/java/com/moyerun/moyeorun_android/login/data/{ => model}/Token.kt (56%) create mode 100644 app/src/main/java/com/moyerun/moyeorun_android/profile/SignUpMetaData.kt diff --git a/app/src/main/AndroidManifest.xml b/app/src/main/AndroidManifest.xml index 6faa503..acf9ee8 100644 --- a/app/src/main/AndroidManifest.xml +++ b/app/src/main/AndroidManifest.xml @@ -13,7 +13,10 @@ android:supportsRtl="true" android:theme="@style/Theme.MoyeoRunandroid"> + { showUnknownErrorToast() - //Todo: #31 을 rebase 하고 주석 풀기 -// Lg.fe("Couldn't get credential from result.", e) + Lg.fe("Couldn't get credential from result.", e) } } } @@ -67,16 +65,16 @@ class LoginActivity : AppCompatActivity() { val binding = ActivityLoginBinding.inflate(layoutInflater) setContentView(binding.root) - observeEvent(viewModel.loginEvent) { - when (it) { - LoginEvent.RegisteredUser -> { - // Todo: 메인화면 진입 + observeEvent(viewModel.loginEvent) { event -> + when (event) { + is LoginEvent.RegisteredUser -> { + // Todo: 메인 화면으로 이동 Lg.d("Login!") } - LoginEvent.NewUser -> { - ProfileEditActivity.startActivity(this) + is LoginEvent.NewUser -> { + ProfileEditActivity.startActivity(this, event.signUpMetaData) } - LoginEvent.Error -> { + is LoginEvent.Error -> { showUnknownErrorToast() } } @@ -91,15 +89,14 @@ class LoginActivity : AppCompatActivity() { beginSignInResultLauncher.launch(intentSenderRequest) } catch (e: IntentSender.SendIntentException) { showUnknownErrorToast() -// Lg.fe("Couldn't start One Tab UI", e) + Lg.fe("Couldn't start One Tab UI", e) } } .addOnFailureListener(this) { // 기기에 등록된 계정이 없는 경우 호출 startDeviceGoogleSignInActivity() - //Todo: #31 을 rebase 하고 주석 풀기 // 간혹 등록된 계정이 있는데도 해당 콜백을 타는 경우가 있어서 로깅 -// Lg.fe("No Google Accounts found", it) + Lg.fe("No Google Accounts found", it) } } } diff --git a/app/src/main/java/com/moyerun/moyeorun_android/login/ui/LoginEvent.kt b/app/src/main/java/com/moyerun/moyeorun_android/login/ui/LoginEvent.kt index f406b21..ecbf333 100644 --- a/app/src/main/java/com/moyerun/moyeorun_android/login/ui/LoginEvent.kt +++ b/app/src/main/java/com/moyerun/moyeorun_android/login/ui/LoginEvent.kt @@ -1,5 +1,9 @@ package com.moyerun.moyeorun_android.login.ui -enum class LoginEvent { - RegisteredUser, NewUser, Error +import com.moyerun.moyeorun_android.profile.SignUpMetaData + +sealed class LoginEvent { + object RegisteredUser : LoginEvent() + data class NewUser(val signUpMetaData: SignUpMetaData) : LoginEvent() + object Error : LoginEvent() } \ No newline at end of file diff --git a/app/src/main/java/com/moyerun/moyeorun_android/login/ui/LoginViewModel.kt b/app/src/main/java/com/moyerun/moyeorun_android/login/ui/LoginViewModel.kt index 824e7fe..ef09a70 100644 --- a/app/src/main/java/com/moyerun/moyeorun_android/login/ui/LoginViewModel.kt +++ b/app/src/main/java/com/moyerun/moyeorun_android/login/ui/LoginViewModel.kt @@ -4,6 +4,8 @@ import androidx.lifecycle.ViewModel import androidx.lifecycle.viewModelScope import com.moyerun.moyeorun_android.common.EventLiveData import com.moyerun.moyeorun_android.common.MutableEventLiveData +import com.moyerun.moyeorun_android.login.ProviderType +import com.moyerun.moyeorun_android.profile.SignUpMetaData import com.moyerun.moyeorun_android.login.data.LoginRepository import dagger.hilt.android.lifecycle.HiltViewModel import kotlinx.coroutines.flow.MutableStateFlow @@ -24,15 +26,16 @@ class LoginViewModel @Inject constructor( val loginEvent: EventLiveData get() = _loginEvent - fun signIn(idToken: String) { + fun googleSignIn(idToken: String) { + signInInternal(idToken, ProviderType.Google) + } + + private fun signInInternal(idToken: String, providerType: ProviderType) { viewModelScope.launch { _isLoading.value = true - // Todo: 반환 타입 결정 후 분기 - // Todo: Firebase crashlytics userId 세팅 - // Todo: SharedPreference 에 유저 메타데이터 세팅 - val result = loginRepository.signIn(idToken) + val result = loginRepository.signIn(idToken, providerType) _loginEvent.event = if (result.isNewUser) { - LoginEvent.NewUser + LoginEvent.NewUser(SignUpMetaData(idToken, providerType)) } else { LoginEvent.RegisteredUser } diff --git a/app/src/main/java/com/moyerun/moyeorun_android/profile/ProfileEditActivity.kt b/app/src/main/java/com/moyerun/moyeorun_android/profile/ProfileEditActivity.kt index 73de13e..2dd10eb 100644 --- a/app/src/main/java/com/moyerun/moyeorun_android/profile/ProfileEditActivity.kt +++ b/app/src/main/java/com/moyerun/moyeorun_android/profile/ProfileEditActivity.kt @@ -8,6 +8,7 @@ import android.widget.EditText import androidx.activity.viewModels import androidx.core.widget.doAfterTextChanged import com.moyerun.moyeorun_android.R +import com.moyerun.moyeorun_android.common.Lg import com.moyerun.moyeorun_android.common.extension.* import com.moyerun.moyeorun_android.databinding.ActivityProfileBinding import dagger.hilt.android.AndroidEntryPoint @@ -26,8 +27,9 @@ class ProfileEditActivity : AppCompatActivity() { val binding = ActivityProfileBinding.inflate(layoutInflater) setContentView(binding.root) + val signUpMetaData: SignUpMetaData? = intent.getParcelableExtra(EXTRA_SIGN_UP_META_DATA) val originalProfile: ProfileUiModel? = intent.getParcelableExtra(EXTRA_PROFILE_UI_MODEL) - val isNewProfile = originalProfile == null + val isNewProfile = (originalProfile == null && signUpMetaData != null) if (isNewProfile) { binding.textviewProfileTitle.text = "기본 정보" @@ -37,7 +39,7 @@ class ProfileEditActivity : AppCompatActivity() { binding.buttonProfileConfirm.text = "완료" } - viewModel.updateProfile(originalProfile) + viewModel.updateData(signUpMetaData, originalProfile) binding.edittextProfileName.doAfterTextChanged { viewModel.onNameChanged(it?.toString().orEmpty()) @@ -60,6 +62,12 @@ class ProfileEditActivity : AppCompatActivity() { } } + binding.buttonProfileConfirm.setOnDebounceClickListener { + if (isNewProfile) { + viewModel.signUp() + } + } + repeatOnStart { launch { viewModel.profileUiModel @@ -86,6 +94,24 @@ class ProfileEditActivity : AppCompatActivity() { } } } + + observeEvent(viewModel.profileEvent) { + when (it) { + is ProfileEvent.SuccessSignUp -> { + // Todo: 환영 액티비티로 이동 + Lg.d("observeEvent : Go to welcome activity!") + } + is ProfileEvent.Error -> { + when (it.error) { + ProfileError.WRONG_ACCESS -> { + Lg.fw("Wrong access. signUpMetadata: $signUpMetaData, originProfile: $originalProfile") + toast("잘못된 접근입니다.") + finish() + } + } + } + } + } } private fun isValidText(text: String): Boolean { @@ -106,11 +132,20 @@ class ProfileEditActivity : AppCompatActivity() { companion object { private const val EXTRA_PROFILE_UI_MODEL = "profileUiModel" + private const val EXTRA_SIGN_UP_META_DATA = "signUpMetaData" - fun startActivity(context: Context, profileUiModel: ProfileUiModel? = null) { + // 프로필 수정 시 사용 + fun startActivity(context: Context, profileUiModel: ProfileUiModel) { context.startActivity(Intent(context, ProfileEditActivity::class.java).apply { putExtra(EXTRA_PROFILE_UI_MODEL, profileUiModel) }) } + + // 회원가입 시 사용 + fun startActivity(context: Context, signUpMetaData: SignUpMetaData) { + context.startActivity(Intent(context, ProfileEditActivity::class.java).apply { + putExtra(EXTRA_SIGN_UP_META_DATA, signUpMetaData) + }) + } } } \ No newline at end of file diff --git a/app/src/main/java/com/moyerun/moyeorun_android/profile/ProfileEditViewModel.kt b/app/src/main/java/com/moyerun/moyeorun_android/profile/ProfileEditViewModel.kt index 2694743..48312fa 100644 --- a/app/src/main/java/com/moyerun/moyeorun_android/profile/ProfileEditViewModel.kt +++ b/app/src/main/java/com/moyerun/moyeorun_android/profile/ProfileEditViewModel.kt @@ -1,6 +1,9 @@ package com.moyerun.moyeorun_android.profile import androidx.lifecycle.ViewModel +import com.moyerun.moyeorun_android.common.EventLiveData +import com.moyerun.moyeorun_android.common.Lg +import com.moyerun.moyeorun_android.common.MutableEventLiveData import kotlinx.coroutines.flow.MutableStateFlow import kotlinx.coroutines.flow.StateFlow import kotlinx.coroutines.flow.update @@ -11,15 +14,30 @@ class ProfileEditViewModel: ViewModel() { val profileUiModel: StateFlow get() = _profileUiModel + private val _profileEvent = MutableEventLiveData() + val profileEvent: EventLiveData + get() = _profileEvent + + private var signUpMetaData: SignUpMetaData? = null private var oldProfileUiModel: ProfileUiModel? = null - private var isNewPost = true + private var isNewProfile = true - fun updateProfile(profileUiModel: ProfileUiModel?) { - if (profileUiModel == null) return - _profileUiModel.update { profileUiModel } - oldProfileUiModel = profileUiModel - isNewPost = false + fun updateData(signUpMetaData: SignUpMetaData?, profileUiModel: ProfileUiModel?) { + when { + signUpMetaData == null && profileUiModel != null -> { + _profileUiModel.update { profileUiModel } + oldProfileUiModel = profileUiModel + isNewProfile = false + } + signUpMetaData != null && profileUiModel == null -> { + this.signUpMetaData = signUpMetaData + isNewProfile = true + } + else -> { + _profileEvent.event = ProfileEvent.Error(ProfileError.WRONG_ACCESS) + } + } } fun onNameChanged(name: String) { @@ -39,4 +57,10 @@ class ProfileEditViewModel: ViewModel() { it.copy(imageUrl = imageUrl) } } + + fun signUp() { + // Todo: 회원가입 API 호출 + Lg.d("SignUp : ${profileUiModel.value}, $signUpMetaData") + _profileEvent.event = ProfileEvent.SuccessSignUp + } } \ No newline at end of file diff --git a/app/src/main/java/com/moyerun/moyeorun_android/profile/ProfileUiModel.kt b/app/src/main/java/com/moyerun/moyeorun_android/profile/ProfileUiModel.kt index 5a35e90..2e79853 100644 --- a/app/src/main/java/com/moyerun/moyeorun_android/profile/ProfileUiModel.kt +++ b/app/src/main/java/com/moyerun/moyeorun_android/profile/ProfileUiModel.kt @@ -8,4 +8,13 @@ data class ProfileUiModel( val imageUrl: String = "", val name: String = "", val nickname: String = "" -): Parcelable \ No newline at end of file +): Parcelable + +sealed class ProfileEvent { + object SuccessSignUp: ProfileEvent() + data class Error(val error: ProfileError): ProfileEvent() +} + +enum class ProfileError { + WRONG_ACCESS +} \ No newline at end of file diff --git a/app/src/main/java/com/moyerun/moyeorun_android/profile/SignUpMetaData.kt b/app/src/main/java/com/moyerun/moyeorun_android/profile/SignUpMetaData.kt new file mode 100644 index 0000000..6481136 --- /dev/null +++ b/app/src/main/java/com/moyerun/moyeorun_android/profile/SignUpMetaData.kt @@ -0,0 +1,16 @@ +package com.moyerun.moyeorun_android.profile + +import android.os.Parcelable +import com.moyerun.moyeorun_android.login.ProviderType +import kotlinx.parcelize.Parcelize + +/** + * 회원가입을 위해 프로필 설정 화면으로 진입 시 + * 회원가입 API 호출에 필요한 메타 데이터를 전달하기 위한 + * DataHolder + */ +@Parcelize +data class SignUpMetaData( + val idToken: String, + val providerType: ProviderType +) : Parcelable