diff --git a/app/src/main/res/xml/path_provider.xml b/app/src/main/res/xml/path_provider.xml index 5b5e8874..e637cf57 100644 --- a/app/src/main/res/xml/path_provider.xml +++ b/app/src/main/res/xml/path_provider.xml @@ -1,4 +1,4 @@ - + \ No newline at end of file diff --git a/data/src/main/java/com/whyranoid/data/datasource/account/AccountDataSourceImpl.kt b/data/src/main/java/com/whyranoid/data/datasource/account/AccountDataSourceImpl.kt index 45f0210e..97bc1f16 100644 --- a/data/src/main/java/com/whyranoid/data/datasource/account/AccountDataSourceImpl.kt +++ b/data/src/main/java/com/whyranoid/data/datasource/account/AccountDataSourceImpl.kt @@ -1,11 +1,16 @@ package com.whyranoid.data.datasource.account import com.whyranoid.data.getResult -import com.whyranoid.data.model.account.ChangeMyInfoRequest import com.whyranoid.data.model.account.SignUpRequest import com.whyranoid.data.model.account.toLoginData +import com.whyranoid.data.model.account.toUserInfo import com.whyranoid.domain.datasource.AccountDataSource import com.whyranoid.domain.model.account.LoginData +import com.whyranoid.domain.model.account.UserInfo +import okhttp3.MediaType +import okhttp3.MultipartBody +import okhttp3.RequestBody +import java.io.File class AccountDataSourceImpl(private val accountService: AccountService) : AccountDataSource { override suspend fun signUp( @@ -54,12 +59,18 @@ class AccountDataSourceImpl(private val accountService: AccountService) : Accoun override suspend fun changeMyInfo(walkieId: Long, nickName: String, profileUrl: String?): Result { return kotlin.runCatching { + var imagePart: MultipartBody.Part? = null + + if (profileUrl != null) { + val file = File(profileUrl) + val fileBody = RequestBody.create(MediaType.parse("image/*"), file) + imagePart = MultipartBody.Part.createFormData("profileImg", file.name, fileBody) + } + val response = accountService.changeMyInfo( walkieId, - ChangeMyInfoRequest( - profileImg = profileUrl ?: "", - nickname = nickName - ) + imagePart, + nickName ) if (response.isSuccessful) { return Result.success(true) @@ -68,4 +79,10 @@ class AccountDataSourceImpl(private val accountService: AccountService) : Accoun } } } + + override suspend fun getUserInfo(walkieId: Long): Result { + return kotlin.runCatching { + accountService.getMyInfo(walkieId).getResult { it.toUserInfo() } + } + } } diff --git a/data/src/main/java/com/whyranoid/data/datasource/account/AccountService.kt b/data/src/main/java/com/whyranoid/data/datasource/account/AccountService.kt index fc7bc7e8..d9af3b92 100644 --- a/data/src/main/java/com/whyranoid/data/datasource/account/AccountService.kt +++ b/data/src/main/java/com/whyranoid/data/datasource/account/AccountService.kt @@ -1,16 +1,18 @@ package com.whyranoid.data.datasource.account import com.whyranoid.data.API -import com.whyranoid.data.model.account.ChangeMyInfoRequest import com.whyranoid.data.model.account.ChangeMyInfoResponse import com.whyranoid.data.model.account.LoginDataResponse import com.whyranoid.data.model.account.NickCheckResponse import com.whyranoid.data.model.account.SignUpRequest import com.whyranoid.data.model.account.SignUpResponse +import com.whyranoid.data.model.account.UserInfoResponse +import okhttp3.MultipartBody import retrofit2.Response import retrofit2.http.Body import retrofit2.http.GET import retrofit2.http.POST +import retrofit2.http.Part import retrofit2.http.Query interface AccountService { @@ -31,7 +33,13 @@ interface AccountService { @POST(API.WalkingControl.MY) suspend fun changeMyInfo( - @Query("walkieId") id: Long, - @Body myInfoRequest: ChangeMyInfoRequest + @Part("walkieId") id: Long, + @Part profileImg: MultipartBody.Part?, + @Part("nickname") nickName: String, ): Response + + @GET(API.WalkingControl.MY) + suspend fun getMyInfo( + @Query("walkieId") id: Long + ): Response } diff --git a/data/src/main/java/com/whyranoid/data/model/account/ChangeMyInfoRequest.kt b/data/src/main/java/com/whyranoid/data/model/account/ChangeMyInfoRequest.kt deleted file mode 100644 index 76f3beda..00000000 --- a/data/src/main/java/com/whyranoid/data/model/account/ChangeMyInfoRequest.kt +++ /dev/null @@ -1,6 +0,0 @@ -package com.whyranoid.data.model.account - -data class ChangeMyInfoRequest ( - val profileImg: String?, - val nickname: String, -) \ No newline at end of file diff --git a/data/src/main/java/com/whyranoid/data/model/account/UserInfoResponse.kt b/data/src/main/java/com/whyranoid/data/model/account/UserInfoResponse.kt new file mode 100644 index 00000000..ae354f3e --- /dev/null +++ b/data/src/main/java/com/whyranoid/data/model/account/UserInfoResponse.kt @@ -0,0 +1,14 @@ +package com.whyranoid.data.model.account + +import com.whyranoid.domain.model.account.UserInfo + +data class UserInfoResponse ( + val nickname: String, + val profileImg: String? +) + +fun UserInfoResponse.toUserInfo() = UserInfo( + name = "", // TODO 수정 + nickname = nickname, + profileImg = profileImg +) \ No newline at end of file diff --git a/data/src/main/java/com/whyranoid/data/repository/AccountRepositoryImpl.kt b/data/src/main/java/com/whyranoid/data/repository/AccountRepositoryImpl.kt index ee1fd41d..e3a466c8 100644 --- a/data/src/main/java/com/whyranoid/data/repository/AccountRepositoryImpl.kt +++ b/data/src/main/java/com/whyranoid/data/repository/AccountRepositoryImpl.kt @@ -4,6 +4,7 @@ import android.util.Log import com.whyranoid.data.AccountDataStore import com.whyranoid.domain.datasource.AccountDataSource import com.whyranoid.domain.model.account.Sex +import com.whyranoid.domain.model.account.UserInfo import com.whyranoid.domain.repository.AccountRepository import kotlinx.coroutines.flow.Flow import kotlinx.coroutines.flow.first @@ -14,13 +15,13 @@ class AccountRepositoryImpl( ) : AccountRepository { override val authId: Flow = accountDataStore.authId - override val uId: Flow = accountDataStore.uId + override val walkieId: Flow = accountDataStore.uId override val userName: Flow = accountDataStore.userName override val nickName: Flow = accountDataStore.nickName override val profileUrl: Flow = accountDataStore.profileUrl override suspend fun getUID(): Long { - return requireNotNull(uId.first()) + return requireNotNull(walkieId.first()) } override suspend fun signUp( @@ -102,4 +103,8 @@ class AccountRepositoryImpl( return Result.failure(Exception("마이페이지 정보 수정 실패")) } } + + override suspend fun getUserInfo(walkieId: Long): Result { + return accountDataSource.getUserInfo(walkieId) + } } diff --git a/domain/src/main/java/com/whyranoid/domain/datasource/AccountDataSource.kt b/domain/src/main/java/com/whyranoid/domain/datasource/AccountDataSource.kt index f2671a75..22ad5805 100644 --- a/domain/src/main/java/com/whyranoid/domain/datasource/AccountDataSource.kt +++ b/domain/src/main/java/com/whyranoid/domain/datasource/AccountDataSource.kt @@ -1,6 +1,7 @@ package com.whyranoid.domain.datasource import com.whyranoid.domain.model.account.LoginData +import com.whyranoid.domain.model.account.UserInfo interface AccountDataSource { suspend fun signUp( @@ -16,4 +17,6 @@ interface AccountDataSource { suspend fun signIn(authorId: String): Result suspend fun changeMyInfo(walkieId: Long, nickName: String, profileUrl: String?): Result + + suspend fun getUserInfo(walkieId: Long): Result } diff --git a/domain/src/main/java/com/whyranoid/domain/model/account/UserInfo.kt b/domain/src/main/java/com/whyranoid/domain/model/account/UserInfo.kt new file mode 100644 index 00000000..7ff45087 --- /dev/null +++ b/domain/src/main/java/com/whyranoid/domain/model/account/UserInfo.kt @@ -0,0 +1,7 @@ +package com.whyranoid.domain.model.account + +data class UserInfo ( + val name: String, + val profileImg: String?, + val nickname: String +) \ No newline at end of file diff --git a/domain/src/main/java/com/whyranoid/domain/repository/AccountRepository.kt b/domain/src/main/java/com/whyranoid/domain/repository/AccountRepository.kt index eb4dc6b7..e89907e2 100644 --- a/domain/src/main/java/com/whyranoid/domain/repository/AccountRepository.kt +++ b/domain/src/main/java/com/whyranoid/domain/repository/AccountRepository.kt @@ -1,12 +1,13 @@ package com.whyranoid.domain.repository import com.whyranoid.domain.model.account.Sex +import com.whyranoid.domain.model.account.UserInfo import kotlinx.coroutines.flow.Flow interface AccountRepository { val authId: Flow - val uId: Flow + val walkieId: Flow val userName: Flow val nickName: Flow val profileUrl: Flow @@ -33,4 +34,6 @@ interface AccountRepository { suspend fun checkNickName(nickName: String): Result> suspend fun changeMyInfo(walkieId: Long, nickName: String, profileUrl: String?): Result + + suspend fun getUserInfo(walkieId: Long): Result } diff --git a/domain/src/main/java/com/whyranoid/domain/usecase/GetChallengePreviewsByTypeUseCase.kt b/domain/src/main/java/com/whyranoid/domain/usecase/GetChallengePreviewsByTypeUseCase.kt index a610fe9e..2c4604a2 100644 --- a/domain/src/main/java/com/whyranoid/domain/usecase/GetChallengePreviewsByTypeUseCase.kt +++ b/domain/src/main/java/com/whyranoid/domain/usecase/GetChallengePreviewsByTypeUseCase.kt @@ -12,7 +12,7 @@ class GetChallengePreviewsByTypeUseCase( ) { suspend operator fun invoke(type: ChallengeType): List { return challengeRepository.getChallengePreviewsByType( - accountRepository.uId.first()?.toInt() ?: -1, type + accountRepository.walkieId.first()?.toInt() ?: -1, type ) } } \ No newline at end of file diff --git a/domain/src/main/java/com/whyranoid/domain/usecase/GetFollowingsPostsUseCase.kt b/domain/src/main/java/com/whyranoid/domain/usecase/GetFollowingsPostsUseCase.kt index b93e9783..4132f493 100644 --- a/domain/src/main/java/com/whyranoid/domain/usecase/GetFollowingsPostsUseCase.kt +++ b/domain/src/main/java/com/whyranoid/domain/usecase/GetFollowingsPostsUseCase.kt @@ -10,7 +10,7 @@ class GetFollowingsPostsUseCase( private val postRepository: PostRepository, ) { suspend operator fun invoke(): Result> { - val myUid = requireNotNull(accountRepository.uId.first()) + val myUid = requireNotNull(accountRepository.walkieId.first()) return postRepository.getMyFollowingsPost(myUid) } } \ No newline at end of file diff --git a/domain/src/main/java/com/whyranoid/domain/usecase/GetMyUidUseCase.kt b/domain/src/main/java/com/whyranoid/domain/usecase/GetMyUidUseCase.kt index d666d6f3..6ebf8cf2 100644 --- a/domain/src/main/java/com/whyranoid/domain/usecase/GetMyUidUseCase.kt +++ b/domain/src/main/java/com/whyranoid/domain/usecase/GetMyUidUseCase.kt @@ -7,6 +7,6 @@ class GetMyUidUseCase( private val accountRepository: AccountRepository, ) { suspend operator fun invoke(): Result { - return runCatching { requireNotNull(accountRepository.uId.firstOrNull()) } + return runCatching { requireNotNull(accountRepository.walkieId.firstOrNull()) } } } diff --git a/domain/src/main/java/com/whyranoid/domain/usecase/LikePostUseCase.kt b/domain/src/main/java/com/whyranoid/domain/usecase/LikePostUseCase.kt index c853ee3a..47e01cd7 100644 --- a/domain/src/main/java/com/whyranoid/domain/usecase/LikePostUseCase.kt +++ b/domain/src/main/java/com/whyranoid/domain/usecase/LikePostUseCase.kt @@ -10,7 +10,7 @@ class LikePostUseCase( ) { suspend operator fun invoke(postId: Long): Result { - val uid = requireNotNull(accountRepository.uId.first()) + val uid = requireNotNull(accountRepository.walkieId.first()) return communityRepository.likePost(postId, uid) } } diff --git a/domain/src/main/java/com/whyranoid/domain/usecase/UploadPostUseCase.kt b/domain/src/main/java/com/whyranoid/domain/usecase/UploadPostUseCase.kt index 93f37791..6f728602 100644 --- a/domain/src/main/java/com/whyranoid/domain/usecase/UploadPostUseCase.kt +++ b/domain/src/main/java/com/whyranoid/domain/usecase/UploadPostUseCase.kt @@ -15,7 +15,7 @@ class UploadPostUseCase @Inject constructor( history: String, imagePath: String, ): Result { - return accountRepository.uId.first()?.let { uid -> + return accountRepository.walkieId.first()?.let { uid -> postRepository.uploadPost(uid, content, colorMode, history, imagePath) } ?: kotlin.run { Result.failure(Exception("Account Error")) diff --git a/domain/src/main/java/com/whyranoid/domain/usecase/community/FollowUseCase.kt b/domain/src/main/java/com/whyranoid/domain/usecase/community/FollowUseCase.kt index 90c82003..9b5a744d 100644 --- a/domain/src/main/java/com/whyranoid/domain/usecase/community/FollowUseCase.kt +++ b/domain/src/main/java/com/whyranoid/domain/usecase/community/FollowUseCase.kt @@ -10,7 +10,7 @@ class FollowUseCase( ) { suspend operator fun invoke(otherUId: Long): Result { return runCatching { - val uid = requireNotNull(accountRepository.uId.first()) + val uid = requireNotNull(accountRepository.walkieId.first()) followRepository.follow(uid, otherUId).getOrThrow() } } diff --git a/domain/src/main/java/com/whyranoid/domain/usecase/community/RemoveFollowerUseCase.kt b/domain/src/main/java/com/whyranoid/domain/usecase/community/RemoveFollowerUseCase.kt index fa650c8a..ffbd4762 100644 --- a/domain/src/main/java/com/whyranoid/domain/usecase/community/RemoveFollowerUseCase.kt +++ b/domain/src/main/java/com/whyranoid/domain/usecase/community/RemoveFollowerUseCase.kt @@ -10,7 +10,7 @@ class RemoveFollowerUseCase( ) { suspend operator fun invoke(otherUId: Long): Result { return runCatching { - val uid = requireNotNull(accountRepository.uId.first()) + val uid = requireNotNull(accountRepository.walkieId.first()) followRepository.unfollow(otherUId, uid).getOrThrow() } } diff --git a/domain/src/main/java/com/whyranoid/domain/usecase/community/SendCommentUseCase.kt b/domain/src/main/java/com/whyranoid/domain/usecase/community/SendCommentUseCase.kt index 78ad3698..cd70ee5d 100644 --- a/domain/src/main/java/com/whyranoid/domain/usecase/community/SendCommentUseCase.kt +++ b/domain/src/main/java/com/whyranoid/domain/usecase/community/SendCommentUseCase.kt @@ -13,7 +13,7 @@ class SendCommentUseCase( postId: Long, content: String, ): Result { - val commenterId = accountRepository.uId.first() ?: return Result.failure(Exception("uid is null")) + val commenterId = accountRepository.walkieId.first() ?: return Result.failure(Exception("uid is null")) return postRepository.sendComment( postId, commenterId, diff --git a/domain/src/main/java/com/whyranoid/domain/usecase/community/UnFollowUseCase.kt b/domain/src/main/java/com/whyranoid/domain/usecase/community/UnFollowUseCase.kt index f9da79b7..1df92466 100644 --- a/domain/src/main/java/com/whyranoid/domain/usecase/community/UnFollowUseCase.kt +++ b/domain/src/main/java/com/whyranoid/domain/usecase/community/UnFollowUseCase.kt @@ -10,7 +10,7 @@ class UnFollowUseCase( ) { suspend operator fun invoke(otherUId: Long): Result { return runCatching { - val uid = requireNotNull(accountRepository.uId.first()) + val uid = requireNotNull(accountRepository.walkieId.first()) followRepository.unfollow(uid, otherUId).getOrThrow() } } diff --git a/domain/src/main/java/com/whyranoid/domain/usecase/running/GetRunningFollowerUseCase.kt b/domain/src/main/java/com/whyranoid/domain/usecase/running/GetRunningFollowerUseCase.kt index 37f08cb4..b2aea021 100644 --- a/domain/src/main/java/com/whyranoid/domain/usecase/running/GetRunningFollowerUseCase.kt +++ b/domain/src/main/java/com/whyranoid/domain/usecase/running/GetRunningFollowerUseCase.kt @@ -17,7 +17,7 @@ class GetRunningFollowerUseCase( var followings = listOf() var runningFollowings = listOf() kotlin.runCatching { - id = requireNotNull(accountRepository.uId.first()) + id = requireNotNull(accountRepository.walkieId.first()) } return callbackFlow { while (true) { diff --git a/domain/src/main/java/com/whyranoid/domain/usecase/running/RunningFinishUseCase.kt b/domain/src/main/java/com/whyranoid/domain/usecase/running/RunningFinishUseCase.kt index 06177d90..0fa0edbb 100644 --- a/domain/src/main/java/com/whyranoid/domain/usecase/running/RunningFinishUseCase.kt +++ b/domain/src/main/java/com/whyranoid/domain/usecase/running/RunningFinishUseCase.kt @@ -9,7 +9,7 @@ class RunningFinishUseCase( private val runningRepository: RunningRepository, ) { suspend operator fun invoke(): Result { - accountRepository.uId.first()?.let { id -> + accountRepository.walkieId.first()?.let { id -> return runningRepository.finishRunning(id) } return Result.failure(Exception("ID 정보 없음")) diff --git a/domain/src/main/java/com/whyranoid/domain/usecase/running/RunningStartUseCase.kt b/domain/src/main/java/com/whyranoid/domain/usecase/running/RunningStartUseCase.kt index 169abf90..d2d74120 100644 --- a/domain/src/main/java/com/whyranoid/domain/usecase/running/RunningStartUseCase.kt +++ b/domain/src/main/java/com/whyranoid/domain/usecase/running/RunningStartUseCase.kt @@ -9,7 +9,7 @@ class RunningStartUseCase( private val runningRepository: RunningRepository, ) { suspend operator fun invoke(): Result { - accountRepository.uId.first()?.let { id -> + accountRepository.walkieId.first()?.let { id -> return runningRepository.startRunning(id) } return Result.failure(Exception("ID 정보 없음")) diff --git a/domain/src/main/java/com/whyranoid/domain/usecase/running/SendLikeUseCase.kt b/domain/src/main/java/com/whyranoid/domain/usecase/running/SendLikeUseCase.kt index 4c19c2d6..c11b6dce 100644 --- a/domain/src/main/java/com/whyranoid/domain/usecase/running/SendLikeUseCase.kt +++ b/domain/src/main/java/com/whyranoid/domain/usecase/running/SendLikeUseCase.kt @@ -10,7 +10,7 @@ class SendLikeUseCase( ) { suspend operator fun invoke(receiverId: Long): Result { return kotlin.runCatching { - val uid = requireNotNull(accountRepository.uId.first()) + val uid = requireNotNull(accountRepository.walkieId.first()) runningRepository.sendLike(uid, receiverId).onSuccess { return Result.success(Unit) }.onFailure { diff --git a/gradle/libs.versions.toml b/gradle/libs.versions.toml index 8c02ccce..331d51f4 100644 --- a/gradle/libs.versions.toml +++ b/gradle/libs.versions.toml @@ -5,7 +5,7 @@ androidx-core = "1.13.1" androidx-appcompat = "1.6.1" androidx-constraintlayout = "2.1.4" naver-maps-android-sdk = "3.16.2" -hilt = "2.44" +hilt = "2.48.1" javax-inject = "1" android-material = "1.8.0" diff --git a/presentation/src/main/java/com/whyranoid/presentation/reusable/SingleToast.kt b/presentation/src/main/java/com/whyranoid/presentation/reusable/SingleToast.kt new file mode 100644 index 00000000..1fa83381 --- /dev/null +++ b/presentation/src/main/java/com/whyranoid/presentation/reusable/SingleToast.kt @@ -0,0 +1,28 @@ +package com.whyranoid.presentation.reusable + +import android.content.Context +import android.widget.Toast + +object SingleToast { + + private const val DEFAULT_DURATION = 2_000L + private var lastToastInfo: Pair? = null + + fun show( + context: Context, + message: String, + duration: Int = Toast.LENGTH_SHORT + ) { + if (lastToastInfo?.first != message) { + lastToastInfo = message to System.currentTimeMillis() + Toast.makeText(context, message, duration).show() + } else { + lastToastInfo?.second?.let {lastToastTime -> + if (System.currentTimeMillis() - lastToastTime > DEFAULT_DURATION) { + lastToastInfo = message to System.currentTimeMillis() + Toast.makeText(context, message, duration).show() + } + } + } + } +} \ No newline at end of file diff --git a/presentation/src/main/java/com/whyranoid/presentation/screens/challenge/ChallengeDetailScreen.kt b/presentation/src/main/java/com/whyranoid/presentation/screens/challenge/ChallengeDetailScreen.kt index 0cbdcc21..cd73490f 100644 --- a/presentation/src/main/java/com/whyranoid/presentation/screens/challenge/ChallengeDetailScreen.kt +++ b/presentation/src/main/java/com/whyranoid/presentation/screens/challenge/ChallengeDetailScreen.kt @@ -1,6 +1,5 @@ package com.whyranoid.presentation.screens.challenge -import android.widget.Toast import androidx.compose.foundation.Image import androidx.compose.foundation.background import androidx.compose.foundation.clickable @@ -45,6 +44,7 @@ import com.whyranoid.presentation.component.ChallengeGoalContent import com.whyranoid.presentation.component.UserIcon import com.whyranoid.presentation.component.bottomsheet.ChallengeExitModalBottomSheetContainer import com.whyranoid.presentation.component.button.WalkiePositiveButton +import com.whyranoid.presentation.reusable.SingleToast import com.whyranoid.presentation.reusable.WalkieCircularProgressIndicator import com.whyranoid.presentation.theme.SystemColor import com.whyranoid.presentation.theme.WalkieTypography @@ -75,12 +75,12 @@ fun ChallengeDetailScreen( viewModel.collectSideEffect { when (it) { ChallengeDetailSideEffect.StartChallengeSuccess -> { - Toast.makeText(context, "챌린지를 성공적으로 시작하였습니다.", Toast.LENGTH_SHORT).show() + SingleToast.show(context, "챌린지를 성공적으로 시작하였습니다.") navController.popBackStack() } ChallengeDetailSideEffect.StartChallengeFailure -> { - Toast.makeText(context, "챌린지를 시작할 수 없습니다.", Toast.LENGTH_SHORT).show() + SingleToast.show(context, "챌린지를 시작할 수 없습니다.") } } } diff --git a/presentation/src/main/java/com/whyranoid/presentation/screens/challenge/ChallengeExitScreen.kt b/presentation/src/main/java/com/whyranoid/presentation/screens/challenge/ChallengeExitScreen.kt index c2638fe1..f9d77257 100644 --- a/presentation/src/main/java/com/whyranoid/presentation/screens/challenge/ChallengeExitScreen.kt +++ b/presentation/src/main/java/com/whyranoid/presentation/screens/challenge/ChallengeExitScreen.kt @@ -1,6 +1,5 @@ package com.whyranoid.presentation.screens.challenge -import android.widget.Toast import androidx.compose.foundation.layout.Arrangement import androidx.compose.foundation.layout.Box import androidx.compose.foundation.layout.Column @@ -11,7 +10,7 @@ import androidx.compose.foundation.layout.height import androidx.compose.foundation.layout.padding import androidx.compose.foundation.layout.size import androidx.compose.foundation.shape.CircleShape -import androidx.compose.material.Text +import androidx.compose.material3.Text import androidx.compose.runtime.Composable import androidx.compose.runtime.LaunchedEffect import androidx.compose.runtime.getValue @@ -27,6 +26,7 @@ import androidx.navigation.NavController import coil.compose.AsyncImage import com.whyranoid.presentation.component.button.WalkieNegativeButton import com.whyranoid.presentation.component.button.WalkiePositiveButton +import com.whyranoid.presentation.reusable.SingleToast import com.whyranoid.presentation.reusable.WalkieCircularProgressIndicator import com.whyranoid.presentation.theme.WalkieTypography import com.whyranoid.presentation.viewmodel.challenge.ChallengeExitSideEffect @@ -54,12 +54,12 @@ fun ChallengeExitScreen( viewModel.collectSideEffect { when (it) { ChallengeExitSideEffect.StopChallengeSuccess -> { - Toast.makeText(context, "챌린지를 성공적으로 중단하였습니다.", Toast.LENGTH_SHORT).show() + SingleToast.show(context, "챌린지를 성공적으로 중단하였습니다.") navController.popBackStack() } ChallengeExitSideEffect.StopChallengeFailure -> { - Toast.makeText(context, "챌린지 중단에 실패하였습니다.", Toast.LENGTH_SHORT).show() + SingleToast.show(context, "챌린지 중단에 실패하였습니다.") } } } diff --git a/presentation/src/main/java/com/whyranoid/presentation/screens/community/CommentScreen.kt b/presentation/src/main/java/com/whyranoid/presentation/screens/community/CommentScreen.kt index fc6c1c97..50e34fa9 100644 --- a/presentation/src/main/java/com/whyranoid/presentation/screens/community/CommentScreen.kt +++ b/presentation/src/main/java/com/whyranoid/presentation/screens/community/CommentScreen.kt @@ -1,7 +1,6 @@ package com.whyranoid.presentation.screens.community import android.util.Log -import android.widget.Toast import androidx.compose.foundation.background import androidx.compose.foundation.clickable import androidx.compose.foundation.layout.Arrangement @@ -19,11 +18,11 @@ import androidx.compose.foundation.layout.wrapContentHeight import androidx.compose.foundation.lazy.LazyColumn import androidx.compose.foundation.shape.RoundedCornerShape import androidx.compose.foundation.text.BasicTextField -import androidx.compose.material.CircularProgressIndicator -import androidx.compose.material.Icon -import androidx.compose.material.LocalTextStyle -import androidx.compose.material.Scaffold -import androidx.compose.material.Text +import androidx.compose.material3.CircularProgressIndicator +import androidx.compose.material3.Icon +import androidx.compose.material3.LocalTextStyle +import androidx.compose.material3.Scaffold +import androidx.compose.material3.Text import androidx.compose.material.icons.Icons import androidx.compose.material.icons.filled.KeyboardArrowLeft import androidx.compose.runtime.Composable @@ -56,6 +55,7 @@ import com.whyranoid.domain.repository.UserRepository import com.whyranoid.domain.usecase.GetMyUidUseCase import com.whyranoid.domain.usecase.community.SendCommentUseCase import com.whyranoid.presentation.component.bar.WalkieTopBar +import com.whyranoid.presentation.reusable.SingleToast import com.whyranoid.presentation.theme.WalkieColor import com.whyranoid.presentation.theme.WalkieTheme import com.whyranoid.presentation.theme.WalkieTypography @@ -128,7 +128,7 @@ fun CommentScreen( isProgress = false }.onFailure { isProgress = false - Toast.makeText(context, "댓글 작성에 실패했습니다.", Toast.LENGTH_SHORT).show() + SingleToast.show(context, "댓글 작성에 실패했습니다.") } } }, diff --git a/presentation/src/main/java/com/whyranoid/presentation/screens/mypage/MyPageScreen.kt b/presentation/src/main/java/com/whyranoid/presentation/screens/mypage/MyPageScreen.kt index 77a46a38..858693ca 100644 --- a/presentation/src/main/java/com/whyranoid/presentation/screens/mypage/MyPageScreen.kt +++ b/presentation/src/main/java/com/whyranoid/presentation/screens/mypage/MyPageScreen.kt @@ -50,6 +50,7 @@ import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.draw.clip import androidx.compose.ui.graphics.Color +import androidx.compose.ui.layout.ContentScale import androidx.compose.ui.text.font.FontWeight import androidx.compose.ui.unit.dp import androidx.compose.ui.unit.sp @@ -84,7 +85,7 @@ fun MyPageScreen( var uid by rememberSaveable { mutableStateOf(null) } LaunchedEffect(Unit) { - val myUid = requireNotNull(viewModel.accountRepository.uId.first()) + val myUid = requireNotNull(viewModel.accountRepository.walkieId.first()) uid = myUid viewModel.getUserDetail(myUid, null) viewModel.getUserBadges(myUid) @@ -198,9 +199,10 @@ fun UserPageContent( AsyncImage( model = userDetail.user.imageUrl, contentDescription = "유저 프로필 이미지", + contentScale = ContentScale.Crop, modifier = Modifier .clip(shape = CircleShape) - .size(70.dp), + .size(64.dp), ) Spacer(modifier = Modifier.width(20.dp)) diff --git a/presentation/src/main/java/com/whyranoid/presentation/screens/mypage/editprofile/EditProfileScreen.kt b/presentation/src/main/java/com/whyranoid/presentation/screens/mypage/editprofile/EditProfileScreen.kt index 792a4b1b..884670ed 100644 --- a/presentation/src/main/java/com/whyranoid/presentation/screens/mypage/editprofile/EditProfileScreen.kt +++ b/presentation/src/main/java/com/whyranoid/presentation/screens/mypage/editprofile/EditProfileScreen.kt @@ -1,7 +1,9 @@ package com.whyranoid.presentation.screens.mypage.editprofile import android.Manifest -import android.widget.Toast +import android.app.Activity +import android.content.Intent +import android.util.Log import androidx.activity.compose.BackHandler import androidx.activity.compose.rememberLauncherForActivityResult import androidx.activity.result.contract.ActivityResultContracts @@ -56,66 +58,68 @@ import androidx.core.content.FileProvider import androidx.lifecycle.compose.collectAsStateWithLifecycle import androidx.navigation.NavController import coil.compose.AsyncImage -import com.whyranoid.domain.util.EMPTY +import coil.request.ImageRequest import com.whyranoid.presentation.R import com.whyranoid.presentation.component.button.CircularIconButton import com.whyranoid.presentation.component.button.WalkieBottomSheetButton import com.whyranoid.presentation.component.button.WalkiePositiveButton +import com.whyranoid.presentation.reusable.SingleToast import com.whyranoid.presentation.reusable.WalkieTextField import com.whyranoid.presentation.theme.WalkieTypography import com.whyranoid.presentation.util.createImageFile import kotlinx.coroutines.flow.collectLatest import kotlinx.coroutines.launch import org.koin.androidx.compose.koinViewModel -import java.util.Objects @OptIn(ExperimentalMaterialApi::class) @Composable fun EditProfileScreen(navController: NavController) { val viewModel = koinViewModel() val walkieId = viewModel.walkieId.collectAsStateWithLifecycle(initialValue = 0L) - val name = viewModel.name.collectAsStateWithLifecycle(initialValue = String.EMPTY) - val nick = viewModel.nick.collectAsStateWithLifecycle(initialValue = String.EMPTY) - val context = LocalContext.current - LaunchedEffect(viewModel.profileImg) { - viewModel.profileImg.collectLatest { - it?.let { url -> viewModel.setProfileUrl(url) } - } - } + val context = LocalContext.current + val coroutineScope = rememberCoroutineScope() val file = context.createImageFile() val uri = FileProvider.getUriForFile( - Objects.requireNonNull(context), - context.packageName + ".provider", file + context, + "com.whyranoid.walkie.provider", + file ) - val cameraLauncher = rememberLauncherForActivityResult(contract = ActivityResultContracts.TakePicture()) { - viewModel.setProfileUrl(uri.toString()) - } + val cameraLauncher = + rememberLauncherForActivityResult(contract = ActivityResultContracts.TakePicture()) { success -> + if (success) { + viewModel.setProfileUrl(uri.toString()) + viewModel.isChangeButtonEnabled.value = true + } + } val cameraPermissionLauncher = rememberLauncherForActivityResult( ActivityResultContracts.RequestPermission() ) { if (it) { - cameraLauncher.launch(uri) + uri?.let { uri -> + cameraLauncher.launch(uri) + } } else { // 권한 거부시 } } val galleryLauncher = rememberLauncherForActivityResult( - ActivityResultContracts.GetContent() - ) { - it?.let { uri -> viewModel.setProfileUrl(uri.toString()) } + contract = ActivityResultContracts.StartActivityForResult() + ) { result -> + if (result.resultCode == Activity.RESULT_OK) { + viewModel.setProfileUrl(result.data?.data.toString()) + viewModel.isChangeButtonEnabled.value = true + } } val bottomSheetState = rememberModalBottomSheetState( initialValue = ModalBottomSheetValue.Hidden ) - val coroutineScope = rememberCoroutineScope() - BackHandler(enabled = bottomSheetState.isVisible) { coroutineScope.launch { bottomSheetState.hide() @@ -143,6 +147,9 @@ fun EditProfileScreen(navController: NavController) { buttonText = "새 프로필 사진 찍기", onClick = { cameraPermissionLauncher.launch(Manifest.permission.CAMERA) + coroutineScope.launch { + bottomSheetState.hide() + } } ) @@ -151,7 +158,14 @@ fun EditProfileScreen(navController: NavController) { WalkieBottomSheetButton( buttonText = "앨범에서 프로필 사진 가져오기", onClick = { - galleryLauncher.launch("image/*") + val intent = Intent(Intent.ACTION_OPEN_DOCUMENT).apply { + addCategory(Intent.CATEGORY_OPENABLE) + type = "image/*" + } + galleryLauncher.launch(intent) + coroutineScope.launch { + bottomSheetState.hide() + } } ) @@ -161,6 +175,9 @@ fun EditProfileScreen(navController: NavController) { buttonText = "현재 프로필 사진 삭제", onClick = { viewModel.setProfileUrl(null) + coroutineScope.launch { + bottomSheetState.hide() + } } ) } @@ -168,8 +185,6 @@ fun EditProfileScreen(navController: NavController) { ) { EditProfileContent( walkieId = walkieId.value ?: 0L, - name = name.value.orEmpty(), - nick = nick.value.orEmpty(), bottomSheetState = bottomSheetState, viewModel = viewModel, ) { @@ -184,8 +199,6 @@ fun EditProfileScreen(navController: NavController) { @Composable fun EditProfileContent( walkieId: Long, - name: String, - nick: String, bottomSheetState: ModalBottomSheetState, viewModel: EditProfileViewModel, popBackStack: () -> Unit @@ -197,31 +210,21 @@ fun EditProfileContent( val focusManager = LocalFocusManager.current val keyboardHeight = WindowInsets.ime.getBottom(LocalDensity.current) val isDuplicateNickName by viewModel.isDuplicateNickName.collectAsStateWithLifecycle() - val currentProfileImg by viewModel.currentProfileUrl.collectAsStateWithLifecycle() - var isChangeEnabled by remember { mutableStateOf(false) } - - var initialProfileImg by remember { mutableStateOf(null) } - LaunchedEffect(viewModel.profileImg) { - viewModel.profileImg.collectLatest { - initialProfileImg = it - } - } + val userInfoUiState by viewModel.userInfoUiState.collectAsStateWithLifecycle() - LaunchedEffect(currentProfileImg) { - if (currentProfileImg != initialProfileImg) { - isChangeEnabled = true - } + LaunchedEffect(Unit) { + viewModel.getUserInfo() } LaunchedEffect(viewModel.isMyInfoChanged) { viewModel.isMyInfoChanged.collectLatest { - Toast.makeText(context, "정보가 수정되었습니다.", Toast.LENGTH_SHORT).show() + SingleToast.show(context, "정보가 수정되었습니다.") popBackStack() } } LaunchedEffect(isDuplicateNickName) { - if (isDuplicateNickName == false) isChangeEnabled = true + if (isDuplicateNickName == false) viewModel.isChangeButtonEnabled.value = true } LaunchedEffect(keyboardHeight) { @@ -230,161 +233,169 @@ fun EditProfileContent( } } - Column( - modifier = Modifier - .fillMaxSize() - .padding(start = 16.dp, end = 16.dp, top = 16.dp) - .verticalScroll(scrollState) - ) { - Box( - Modifier.fillMaxSize() - ) { - Text( - modifier = Modifier.align(Alignment.Center), - text = "프로필 편집", - style = WalkieTypography.Title - ) - Icon( - imageVector = Icons.Filled.Close, - contentDescription = "close button", - modifier = Modifier - .size(24.dp) - .align(Alignment.CenterStart) - .clickable { - popBackStack() - }, - ) - } - - Spacer(modifier = Modifier.height(48.dp)) + if (userInfoUiState != null) { + val name by remember { mutableStateOf(userInfoUiState?.name) } + var nickname by remember { mutableStateOf(userInfoUiState?.nickname) } + val profileImg = userInfoUiState?.profileUrl - Box( + Column( modifier = Modifier - .align(Alignment.CenterHorizontally) - .size(90.dp), + .fillMaxSize() + .padding(start = 16.dp, end = 16.dp, top = 16.dp) + .verticalScroll(scrollState) ) { - AsyncImage( - model = currentProfileImg, - contentDescription = "프로필 이미지", - contentScale = ContentScale.Crop, - modifier = Modifier - .matchParentSize() - .clip(shape = CircleShape) - ) - - CircularIconButton( - modifier = Modifier.align(Alignment.BottomEnd), - onClick = { - coroutineScope.launch { - bottomSheetState.show() - } - }, - contentDescription = "프로필 편집", - icon = ImageVector.vectorResource(id = R.drawable.ic_edit_icon), - tint = Color(0xFF999999), - backgroundColor = Color(0xFFEEEEEE), - iconSize = 24.dp, - buttonSize = 30.dp, - ) - } + Box( + Modifier.fillMaxSize() + ) { + Text( + modifier = Modifier.align(Alignment.Center), + text = "프로필 편집", + style = WalkieTypography.Title + ) + Icon( + imageVector = Icons.Filled.Close, + contentDescription = "close button", + modifier = Modifier + .size(24.dp) + .align(Alignment.CenterStart) + .clickable { + popBackStack() + }, + ) + } - Spacer(modifier = Modifier.height(22.dp)) + Spacer(modifier = Modifier.height(48.dp)) - Text( - modifier = Modifier.align(Alignment.CenterHorizontally), - text = "프로필 사진 변경", - style = WalkieTypography.Body1_SemiBold - ) + Box( + modifier = Modifier + .align(Alignment.CenterHorizontally) + .size(90.dp), + ) { + AsyncImage( + model = ImageRequest.Builder(context) + .data(profileImg) + .fallback(R.drawable.ic_default_profile) + .error(R.drawable.ic_default_profile) + .build(), + onError = { + Log.d("sm.shin", "error: ${it.result.throwable.message}") + }, + contentDescription = "프로필 이미지", + contentScale = ContentScale.Crop, + modifier = Modifier + .size(90.dp) + .clip(shape = CircleShape) + ) - Spacer(modifier = Modifier.height(25.dp)) + CircularIconButton( + modifier = Modifier.align(Alignment.BottomEnd), + onClick = { + coroutineScope.launch { + bottomSheetState.show() + } + }, + contentDescription = "프로필 편집", + icon = ImageVector.vectorResource(id = R.drawable.ic_edit_icon), + tint = Color(0xFF999999), + backgroundColor = Color(0xFFEEEEEE), + iconSize = 24.dp, + buttonSize = 30.dp, + ) + } - Text( - modifier = Modifier.align(Alignment.Start), - text = "이름", - style = WalkieTypography.Body1 - ) + Spacer(modifier = Modifier.height(22.dp)) - Spacer(modifier = Modifier.height(10.dp)) + Text( + modifier = Modifier.align(Alignment.CenterHorizontally), + text = "프로필 사진 변경", + style = WalkieTypography.Body1_SemiBold + ) - WalkieTextField( - text = name, - readOnly = true, - focusRequester = focusRequester, - ) + Spacer(modifier = Modifier.height(25.dp)) - Spacer(modifier = Modifier.height(20.dp)) + Text( + modifier = Modifier.align(Alignment.Start), + text = "이름", + style = WalkieTypography.Body1 + ) - Text( - modifier = Modifier.align(Alignment.Start), - text = "워키닉네임", - style = WalkieTypography.Body1 - ) + Spacer(modifier = Modifier.height(10.dp)) - Spacer(modifier = Modifier.height(10.dp)) + WalkieTextField( + text = name.orEmpty(), + readOnly = true, + focusRequester = focusRequester, + ) - var nickName by remember { mutableStateOf(nick) } - var isEditMode by remember { mutableStateOf(false) } + Spacer(modifier = Modifier.height(20.dp)) - LaunchedEffect(nick) { - nickName = nick - } + Text( + modifier = Modifier.align(Alignment.Start), + text = "워키닉네임", + style = WalkieTypography.Body1 + ) - val nickNameRegex = Regex("[a-zA-Z0-9_.]{0,29}") - - WalkieTextField( - modifier = Modifier, - focusRequester = focusRequester, - text = nickName, - isEnabled = isEditMode, - isValidValue = isDuplicateNickName?.not(), - trailings = { - if (isEditMode.not()) { - Image( - painter = painterResource(id = R.drawable.ic_edit_icon), - contentDescription = "edit icon", - modifier = Modifier - .clickable { - focusRequester.requestFocus() - isEditMode = true - } - ) - } else { - Text( - text = "확인", - style = WalkieTypography.Caption.copy( - fontWeight = FontWeight.Bold - ), - modifier = Modifier - .clickable { - viewModel.checkDuplicateNickName(nickName) - // 성공 시 editMode 해제 - focusManager.clearFocus() - } - ) - } + Spacer(modifier = Modifier.height(10.dp)) + + var isEditMode by remember { mutableStateOf(false) } + + val nickNameRegex = Regex("[a-zA-Z0-9_.]{0,29}") + + WalkieTextField( + modifier = Modifier, + focusRequester = focusRequester, + text = nickname.orEmpty(), + isEnabled = isEditMode, + isValidValue = isDuplicateNickName?.not(), + trailings = { + if (isEditMode.not()) { + Image( + painter = painterResource(id = R.drawable.ic_edit_icon), + contentDescription = "edit icon", + modifier = Modifier + .clickable { + focusRequester.requestFocus() + isEditMode = true + } + ) + } else { + Text( + text = "확인", + style = WalkieTypography.Caption.copy( + fontWeight = FontWeight.Bold + ), + modifier = Modifier + .clickable { + viewModel.checkDuplicateNickName(nickname.orEmpty()) + // 성공 시 editMode 해제 + focusManager.clearFocus() + } + ) + } - }, - onValueChange = { - if (it.matches(nickNameRegex)) { - nickName = it - } else { - Toast.makeText(context, "닉네임은 30자 이내로 영문,숫자,마침표,_만 입력해주세요.", Toast.LENGTH_SHORT).show() + }, + onValueChange = { + if (it.matches(nickNameRegex)) { + nickname = it + } else { + SingleToast.show(context, "닉네임은 30자 이내로 영문,숫자,마침표,_만 입력해주세요.") + } } - } - ) + ) - Spacer(modifier = Modifier.height(40.dp)) + Spacer(modifier = Modifier.height(40.dp)) - Spacer(modifier = Modifier.weight(1f)) + Spacer(modifier = Modifier.weight(1f)) - WalkiePositiveButton( - text = "변경", - isEnabled = isChangeEnabled, - onClicked = { - viewModel.changeMyInfo(walkieId, nickName, currentProfileImg) - } - ) + WalkiePositiveButton( + text = "변경", + isEnabled = viewModel.isChangeButtonEnabled.value, + onClicked = { + viewModel.changeMyInfo(walkieId, nickname.orEmpty(), profileImg) + } + ) - Spacer(modifier = Modifier.height(20.dp)) + Spacer(modifier = Modifier.height(20.dp)) + } } } \ No newline at end of file diff --git a/presentation/src/main/java/com/whyranoid/presentation/screens/mypage/editprofile/EditProfileViewModel.kt b/presentation/src/main/java/com/whyranoid/presentation/screens/mypage/editprofile/EditProfileViewModel.kt index 32d0f426..8a879433 100644 --- a/presentation/src/main/java/com/whyranoid/presentation/screens/mypage/editprofile/EditProfileViewModel.kt +++ b/presentation/src/main/java/com/whyranoid/presentation/screens/mypage/editprofile/EditProfileViewModel.kt @@ -1,5 +1,6 @@ package com.whyranoid.presentation.screens.mypage.editprofile +import androidx.compose.runtime.mutableStateOf import androidx.lifecycle.ViewModel import androidx.lifecycle.viewModelScope import com.whyranoid.domain.repository.AccountRepository @@ -8,6 +9,7 @@ import kotlinx.coroutines.flow.MutableSharedFlow import kotlinx.coroutines.flow.MutableStateFlow import kotlinx.coroutines.flow.asSharedFlow import kotlinx.coroutines.flow.asStateFlow +import kotlinx.coroutines.flow.firstOrNull import kotlinx.coroutines.flow.update import kotlinx.coroutines.launch @@ -21,13 +23,29 @@ class EditProfileViewModel( private val _isMyInfoChanged = MutableSharedFlow() val isMyInfoChanged = _isMyInfoChanged.asSharedFlow() - private val _currentProfileUrl = MutableStateFlow(null) - val currentProfileUrl = _currentProfileUrl.asStateFlow() + private val _userInfoUiState = MutableStateFlow(null) + val userInfoUiState = _userInfoUiState.asStateFlow() - val name = accountRepository.userName - val nick = accountRepository.nickName - val profileImg = accountRepository.profileUrl - val walkieId = accountRepository.uId + val isChangeButtonEnabled = mutableStateOf(false) + + val walkieId = accountRepository.walkieId + + fun getUserInfo() { + viewModelScope.launch(Dispatchers.IO) { + val walkieId = walkieId.firstOrNull() ?: return@launch + + accountRepository.getUserInfo(walkieId) + .onSuccess { userInfo -> + _userInfoUiState.update { + UserInfoUiState( + userInfo.name, + userInfo.nickname, + userInfo.profileImg + ) + } + } + } + } fun checkDuplicateNickName(nickName: String) = viewModelScope.launch(Dispatchers.IO) { accountRepository.checkNickName(nickName) @@ -37,12 +55,16 @@ class EditProfileViewModel( .onFailure { _isDuplicateNickName.update { null } } } - fun changeMyInfo(walkieId: Long, nickName: String, profileUrl: String?) = viewModelScope.launch(Dispatchers.IO) { - accountRepository.changeMyInfo(walkieId, nickName, profileUrl) - .onSuccess { _isMyInfoChanged.emit(true) } - } + fun changeMyInfo(walkieId: Long, nickName: String, profileUrl: String?) = + viewModelScope.launch(Dispatchers.IO) { + accountRepository.changeMyInfo(walkieId, nickName, profileUrl) + .onSuccess { + _isMyInfoChanged.emit(true) + _userInfoUiState.update { it?.copy(nickname = nickName, profileUrl = profileUrl) } + } + } fun setProfileUrl(url: String?) { - _currentProfileUrl.update { url } + _userInfoUiState.update { it?.copy(profileUrl = url) } } } diff --git a/presentation/src/main/java/com/whyranoid/presentation/screens/mypage/editprofile/UserInfoUiState.kt b/presentation/src/main/java/com/whyranoid/presentation/screens/mypage/editprofile/UserInfoUiState.kt new file mode 100644 index 00000000..65e52360 --- /dev/null +++ b/presentation/src/main/java/com/whyranoid/presentation/screens/mypage/editprofile/UserInfoUiState.kt @@ -0,0 +1,7 @@ +package com.whyranoid.presentation.screens.mypage.editprofile + +data class UserInfoUiState( + val name: String, + val nickname: String, + val profileUrl: String? +) \ No newline at end of file diff --git a/presentation/src/main/java/com/whyranoid/presentation/screens/setting/SettingScreen.kt b/presentation/src/main/java/com/whyranoid/presentation/screens/setting/SettingScreen.kt index 7a184bd8..6492deee 100644 --- a/presentation/src/main/java/com/whyranoid/presentation/screens/setting/SettingScreen.kt +++ b/presentation/src/main/java/com/whyranoid/presentation/screens/setting/SettingScreen.kt @@ -28,6 +28,7 @@ import androidx.compose.runtime.collectAsState import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.draw.clip +import androidx.compose.ui.layout.ContentScale import androidx.compose.ui.text.style.TextAlign import androidx.compose.ui.unit.dp import androidx.navigation.NavHostController @@ -98,6 +99,7 @@ fun ProfileSection( AsyncImage( model = user.imageUrl, contentDescription = "Profile Image", + contentScale = ContentScale.Crop, modifier = Modifier .size(64.dp) .clip(CircleShape) diff --git a/presentation/src/main/java/com/whyranoid/presentation/screens/signin/SignInInitialScreen.kt b/presentation/src/main/java/com/whyranoid/presentation/screens/signin/SignInInitialScreen.kt index e6ff3d22..d8f87602 100644 --- a/presentation/src/main/java/com/whyranoid/presentation/screens/signin/SignInInitialScreen.kt +++ b/presentation/src/main/java/com/whyranoid/presentation/screens/signin/SignInInitialScreen.kt @@ -21,7 +21,7 @@ import androidx.compose.foundation.layout.padding import androidx.compose.foundation.layout.size import androidx.compose.foundation.shape.RoundedCornerShape import androidx.compose.material.Scaffold -import androidx.compose.material.Text +import androidx.compose.material3.Text import androidx.compose.material.rememberScaffoldState import androidx.compose.runtime.Composable import androidx.compose.runtime.rememberCoroutineScope @@ -41,6 +41,7 @@ import com.google.android.gms.common.api.ApiException import com.google.android.gms.tasks.Task import com.whyranoid.domain.usecase.RequestLoginUseCase import com.whyranoid.presentation.R +import com.whyranoid.presentation.reusable.SingleToast import com.whyranoid.presentation.theme.WalkieTheme import com.whyranoid.presentation.theme.WalkieTypography import kotlinx.coroutines.launch @@ -55,6 +56,7 @@ fun SignInInitialScreen( ) { val scope = rememberCoroutineScope() val scaffoldState = rememberScaffoldState() + val context = LocalContext.current val requestLoginUseCase = get() @@ -79,18 +81,14 @@ fun SignInInitialScreen( account = account, goToAgreeState = goToAgreeState, ) { errorMsg -> - scope.launch { - scaffoldState.snackbarHostState.showSnackbar(errorMsg) - } + SingleToast.show(context, errorMsg) } } } } } else { - scope.launch { - scaffoldState.snackbarHostState.showSnackbar("계정 연결에 실패했습니다.") - } + SingleToast.show(context, "계정 연결에 실패했습니다.") } } @@ -111,7 +109,6 @@ fun SignInInitialScreen( .padding(bottom = 232.dp), ) - val context = LocalContext.current Box( modifier = Modifier .align(Alignment.BottomCenter) @@ -151,12 +148,12 @@ fun SignInInitialScreen( private fun handleSignInResult( account: GoogleSignInAccount, goToAgreeState: (authId: String, name: String, url: String?) -> Unit, - showErrorSnackBar: (msg: String) -> Unit, + showErrorToast: (String) -> Unit ) { runCatching { - val uid = requireNotNull(account.id) { showErrorSnackBar("authId is null") } - val name = requireNotNull(account.displayName) { showErrorSnackBar("name is null") } - val url = account.photoUrl?.let { it.toString() } + val uid = requireNotNull(account.id) { showErrorToast("authId is null") } + val name = requireNotNull(account.displayName) { showErrorToast("name is null") } + val url = account.photoUrl?.toString() goToAgreeState(uid, name, url) } } diff --git a/presentation/src/main/java/com/whyranoid/presentation/util/extensions.kt b/presentation/src/main/java/com/whyranoid/presentation/util/extensions.kt index d86de8dc..a7bd5935 100644 --- a/presentation/src/main/java/com/whyranoid/presentation/util/extensions.kt +++ b/presentation/src/main/java/com/whyranoid/presentation/util/extensions.kt @@ -6,6 +6,7 @@ import android.content.Context import android.content.Intent import android.net.Uri import android.os.Build +import android.os.Environment import android.provider.Settings import androidx.annotation.RequiresApi import java.io.File @@ -80,11 +81,11 @@ fun Activity.openStatusBar() { fun Context.createImageFile(): File { val timeStamp = SimpleDateFormat("yyyyMMdd_HHmmss", Locale.getDefault()).format(Date()) - val imageFileName = "JPEG_" + timeStamp + "_" - val image = File.createTempFile( + val storageDir: File? = getExternalFilesDir(Environment.DIRECTORY_PICTURES) + val imageFileName = "JPEG_${timeStamp}_" + return File.createTempFile( imageFileName, ".jpg", - externalCacheDir + storageDir ) - return image } \ No newline at end of file diff --git a/presentation/src/main/java/com/whyranoid/presentation/viewmodel/SplashViewModel.kt b/presentation/src/main/java/com/whyranoid/presentation/viewmodel/SplashViewModel.kt index 16338938..284be9ad 100644 --- a/presentation/src/main/java/com/whyranoid/presentation/viewmodel/SplashViewModel.kt +++ b/presentation/src/main/java/com/whyranoid/presentation/viewmodel/SplashViewModel.kt @@ -24,7 +24,7 @@ class SplashViewModel(private val accountRepository: AccountRepository) : ViewMo _splashState.value = if (isSignedIn()) SplashState.SignedInState else SplashState.SignInState - accountRepository.uId.collect { + accountRepository.walkieId.collect { if (it == null) { _splashState.value = SplashState.SignInState } @@ -37,7 +37,7 @@ class SplashViewModel(private val accountRepository: AccountRepository) : ViewMo } private suspend fun isSignedIn(): Boolean { - return accountRepository.uId.first() != null + return accountRepository.walkieId.first() != null } } diff --git a/presentation/src/main/res/drawable/ic_default_profile.xml b/presentation/src/main/res/drawable/ic_default_profile.xml new file mode 100644 index 00000000..eb718bbe --- /dev/null +++ b/presentation/src/main/res/drawable/ic_default_profile.xml @@ -0,0 +1,25 @@ + + + + + + + + + +