diff --git a/app/build.gradle.kts b/app/build.gradle.kts index e0e57ae5..d9e85826 100644 --- a/app/build.gradle.kts +++ b/app/build.gradle.kts @@ -57,6 +57,7 @@ dependencies { projects.core.domain, projects.core.network, projects.core.ui, + projects.core.util, projects.feature.complete, projects.feature.home, projects.feature.onboarding, diff --git a/core/data/src/main/kotlin/com/nexters/bandalart/android/core/data/di/ServiceModule.kt b/core/data/src/main/kotlin/com/nexters/bandalart/android/core/data/di/ServiceModule.kt index 42e051d8..57f5a1da 100644 --- a/core/data/src/main/kotlin/com/nexters/bandalart/android/core/data/di/ServiceModule.kt +++ b/core/data/src/main/kotlin/com/nexters/bandalart/android/core/data/di/ServiceModule.kt @@ -2,13 +2,14 @@ package com.nexters.bandalart.android.core.data.di import com.nexters.bandalart.android.core.data.service.BandalartService import com.nexters.bandalart.android.core.data.service.GuestLoginService +import com.nexters.bandalart.android.core.network.di.BandalartApi +import com.nexters.bandalart.android.core.network.di.LoginApi import dagger.Module import dagger.Provides import dagger.hilt.InstallIn import dagger.hilt.components.SingletonComponent -import javax.inject.Singleton import retrofit2.Retrofit -import javax.inject.Named +import javax.inject.Singleton @Module @InstallIn(SingletonComponent::class) @@ -17,8 +18,7 @@ internal object ServiceModule { @Singleton @Provides internal fun provideBandalartService( - @Named("HttpClient") - retrofit: Retrofit, + @BandalartApi retrofit: Retrofit, ): BandalartService { return retrofit.create(BandalartService::class.java) } @@ -26,8 +26,7 @@ internal object ServiceModule { @Singleton @Provides internal fun provideGuestLoginService( - @Named("AuthHttpClient") - retrofit: Retrofit, + @LoginApi retrofit: Retrofit, ): GuestLoginService { return retrofit.create(GuestLoginService::class.java) } diff --git a/core/network/src/main/kotlin/com/nexters/bandalart/android/core/network/di/NetworkModule.kt b/core/network/src/main/kotlin/com/nexters/bandalart/android/core/network/di/NetworkModule.kt index 9b530f8e..cffdd6c8 100644 --- a/core/network/src/main/kotlin/com/nexters/bandalart/android/core/network/di/NetworkModule.kt +++ b/core/network/src/main/kotlin/com/nexters/bandalart/android/core/network/di/NetworkModule.kt @@ -19,8 +19,6 @@ import io.ktor.client.request.header import io.ktor.http.ContentType import io.ktor.http.contentType import io.ktor.serialization.kotlinx.json.json -import java.util.concurrent.TimeUnit -import javax.inject.Singleton import kotlinx.coroutines.runBlocking import kotlinx.serialization.json.Json import okhttp3.Interceptor @@ -29,7 +27,8 @@ import okhttp3.OkHttpClient import okhttp3.logging.HttpLoggingInterceptor import retrofit2.Retrofit import timber.log.Timber -import javax.inject.Named +import java.util.concurrent.TimeUnit +import javax.inject.Singleton private const val MaxTimeoutMillis = 3000L private const val MaxRetryCount = 3 @@ -91,10 +90,10 @@ internal object NetworkModule { } } + @LoginApi @Singleton @Provides - @Named("AuthHttpClient") - internal fun provideRetrofitAuthHttpClient( + internal fun provideLoginApiRetrofit( httpLoggingInterceptor: HttpLoggingInterceptor, ): Retrofit { val contentType = "application/json".toMediaType() @@ -110,10 +109,10 @@ internal object NetworkModule { .build() } + @BandalartApi @Singleton @Provides - @Named("HttpClient") - internal fun provideRetrofitHttpClient( + internal fun provideBandalartApiRetrofit( dataStoreProvider: DataStoreProvider, httpLoggingInterceptor: HttpLoggingInterceptor, ): Retrofit { diff --git a/core/network/src/main/kotlin/com/nexters/bandalart/android/core/network/di/RetrofitQualifier.kt b/core/network/src/main/kotlin/com/nexters/bandalart/android/core/network/di/RetrofitQualifier.kt new file mode 100644 index 00000000..517a1a83 --- /dev/null +++ b/core/network/src/main/kotlin/com/nexters/bandalart/android/core/network/di/RetrofitQualifier.kt @@ -0,0 +1,11 @@ +package com.nexters.bandalart.android.core.network.di + +import javax.inject.Qualifier + +@Qualifier +@Retention(AnnotationRetention.BINARY) +annotation class BandalartApi + +@Qualifier +@Retention(AnnotationRetention.BINARY) +annotation class LoginApi diff --git a/core/ui/build.gradle.kts b/core/ui/build.gradle.kts index e3500377..024286c8 100644 --- a/core/ui/build.gradle.kts +++ b/core/ui/build.gradle.kts @@ -16,6 +16,7 @@ android { dependencies { implementations( projects.core.designsystem, + projects.core.util, libs.kotlinx.datetime, libs.androidx.core, libs.lottie.compose, diff --git a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/extension/MultipleEventsCutter.kt b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/MultipleEventsCutter.kt similarity index 90% rename from core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/extension/MultipleEventsCutter.kt rename to core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/MultipleEventsCutter.kt index 66c1945c..9d12d8c4 100644 --- a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/extension/MultipleEventsCutter.kt +++ b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/MultipleEventsCutter.kt @@ -1,4 +1,4 @@ -package com.nexters.bandalart.android.core.ui.extension +package com.nexters.bandalart.android.core.ui internal interface MultipleEventsCutter { fun processEvent(event: () -> Unit) diff --git a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/ObserveEvent.kt b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/ObserveEvent.kt new file mode 100644 index 00000000..fc7b78a4 --- /dev/null +++ b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/ObserveEvent.kt @@ -0,0 +1,23 @@ +package com.nexters.bandalart.android.core.ui + +import androidx.compose.runtime.Composable +import androidx.compose.runtime.LaunchedEffect +import androidx.compose.ui.platform.LocalLifecycleOwner +import androidx.lifecycle.Lifecycle +import androidx.lifecycle.repeatOnLifecycle +import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.flow.Flow +import kotlinx.coroutines.withContext + +// https://www.youtube.com/watch?v=njchj9d_Lf8&t=1218s +@Composable +fun ObserveAsEvents(flow: Flow, onEvent: (T) -> Unit) { + val lifecycleOwner = LocalLifecycleOwner.current + LaunchedEffect(flow, lifecycleOwner.lifecycle) { + lifecycleOwner.repeatOnLifecycle(Lifecycle.State.STARTED) { + withContext(Dispatchers.Main.immediate) { + flow.collect(onEvent) + } + } + } +} diff --git a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/extension/TextUnit.kt b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/TextUnit.kt similarity index 84% rename from core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/extension/TextUnit.kt rename to core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/TextUnit.kt index cdcd9d5a..f779c937 100644 --- a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/extension/TextUnit.kt +++ b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/TextUnit.kt @@ -1,4 +1,4 @@ -package com.nexters.bandalart.android.core.ui.extension +package com.nexters.bandalart.android.core.ui import androidx.compose.runtime.Composable import androidx.compose.ui.platform.LocalDensity diff --git a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/extension/ThemeColor.kt b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/ThemeColor.kt similarity index 84% rename from core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/extension/ThemeColor.kt rename to core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/ThemeColor.kt index 32e534d2..6e2410c9 100644 --- a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/extension/ThemeColor.kt +++ b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/ThemeColor.kt @@ -1,4 +1,4 @@ -package com.nexters.bandalart.android.core.ui.extension +package com.nexters.bandalart.android.core.ui data class ThemeColor( val mainColor: String, diff --git a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/UiText.kt b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/UiText.kt new file mode 100644 index 00000000..a1d80117 --- /dev/null +++ b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/UiText.kt @@ -0,0 +1,26 @@ +package com.nexters.bandalart.android.core.ui + +import android.content.Context +import androidx.annotation.StringRes +import androidx.compose.runtime.Composable +import androidx.compose.ui.res.stringResource + +sealed class UiText { + data class DirectString(val value: String) : UiText() + + class StringResource( + @StringRes val resId: Int, + vararg val args: Any, + ) : UiText() + + @Composable + fun asString() = when (this) { + is DirectString -> value + is StringResource -> stringResource(resId, *args) + } + + fun asString(context: Context) = when (this) { + is DirectString -> value + is StringResource -> context.getString(resId, *args) + } +} diff --git a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/extension/WindowInsets.kt b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/WindowInsets.kt similarity index 72% rename from core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/extension/WindowInsets.kt rename to core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/WindowInsets.kt index 1de827e9..8a61a266 100644 --- a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/extension/WindowInsets.kt +++ b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/WindowInsets.kt @@ -1,4 +1,4 @@ -package com.nexters.bandalart.android.core.ui.extension +package com.nexters.bandalart.android.core.ui import android.app.Activity import android.graphics.Rect @@ -7,6 +7,8 @@ import androidx.compose.foundation.layout.systemBars import androidx.compose.runtime.Composable import androidx.compose.ui.platform.LocalContext import androidx.compose.ui.platform.LocalDensity +import androidx.compose.ui.unit.Dp +import androidx.compose.ui.unit.dp // https://sungbin.land/jetpack-compose-windowinsets-fa8f286f092b val NavigationBarHeightDp @@ -33,3 +35,9 @@ val StatusBarHeightDp rectangle.top.toDp() } } + +// https://stackoverflow.com/questions/75123079/how-do-i-detect-which-type-of-navigation-bar-is-active +@Composable +fun getNavigationBarPadding(): Dp { + return if (NavigationBarHeightDp == 0.dp) 32.dp else NavigationBarHeightDp - 16.dp +} diff --git a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/BandalartButton.kt b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/BandalartButton.kt index 73c21875..d047cf82 100644 --- a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/BandalartButton.kt +++ b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/BandalartButton.kt @@ -15,7 +15,7 @@ import androidx.compose.ui.text.font.FontWeight import androidx.compose.ui.unit.dp import androidx.compose.ui.unit.sp import com.nexters.bandalart.android.core.ui.extension.clickableSingle -import com.nexters.bandalart.android.core.ui.extension.nonScaleSp +import com.nexters.bandalart.android.core.ui.nonScaleSp import com.nexters.bandalart.android.core.designsystem.theme.Gray900 import com.nexters.bandalart.android.core.designsystem.theme.White import com.nexters.bandalart.android.core.designsystem.theme.pretendard diff --git a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/BandalartDeleteAlertDialog.kt b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/BandalartDeleteAlertDialog.kt index 26288183..9a9f0210 100644 --- a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/BandalartDeleteAlertDialog.kt +++ b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/BandalartDeleteAlertDialog.kt @@ -25,7 +25,7 @@ import androidx.compose.ui.unit.dp import androidx.compose.ui.unit.sp import androidx.compose.ui.window.Dialog import com.nexters.bandalart.android.core.ui.R -import com.nexters.bandalart.android.core.ui.extension.nonScaleSp +import com.nexters.bandalart.android.core.ui.nonScaleSp import com.nexters.bandalart.android.core.designsystem.theme.Gray200 import com.nexters.bandalart.android.core.designsystem.theme.Gray400 import com.nexters.bandalart.android.core.designsystem.theme.Gray900 @@ -33,11 +33,11 @@ import com.nexters.bandalart.android.core.designsystem.theme.White @Composable fun BandalartDeleteAlertDialog( - modifier: Modifier = Modifier, title: String, message: String, onDeleteClicked: () -> Unit, onCancelClicked: () -> Unit, + modifier: Modifier = Modifier, ) { val context = LocalContext.current Dialog(onDismissRequest = { onCancelClicked() }) { diff --git a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/BandalartDropDownMenu.kt b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/BandalartDropDownMenu.kt index 14440eb4..6fb32843 100644 --- a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/BandalartDropDownMenu.kt +++ b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/BandalartDropDownMenu.kt @@ -23,17 +23,17 @@ import androidx.compose.ui.unit.DpOffset import androidx.compose.ui.unit.dp import androidx.compose.ui.unit.sp import com.nexters.bandalart.android.core.ui.R -import com.nexters.bandalart.android.core.ui.extension.nonScaleSp +import com.nexters.bandalart.android.core.ui.nonScaleSp import com.nexters.bandalart.android.core.designsystem.theme.Error import com.nexters.bandalart.android.core.designsystem.theme.White import com.nexters.bandalart.android.core.designsystem.theme.pretendard @Composable fun BandalartDropDownMenu( - modifier: Modifier = Modifier, openDropDownMenu: (Boolean) -> Unit, isDropDownMenuOpened: Boolean, onDeleteClicked: () -> Unit, + modifier: Modifier = Modifier, ) { val context = LocalContext.current MaterialTheme( diff --git a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/CellText.kt b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/CellText.kt index 8e18d6b2..0d374dce 100644 --- a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/CellText.kt +++ b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/CellText.kt @@ -12,7 +12,7 @@ import androidx.compose.ui.text.style.LineBreak import androidx.compose.ui.text.style.TextAlign import androidx.compose.ui.unit.dp import androidx.compose.ui.unit.sp -import com.nexters.bandalart.android.core.ui.extension.nonScaleSp +import com.nexters.bandalart.android.core.ui.nonScaleSp import com.nexters.bandalart.android.core.designsystem.theme.pretendard val cellLineBreak = LineBreak( @@ -23,10 +23,10 @@ val cellLineBreak = LineBreak( @Composable fun CellText( - modifier: Modifier = Modifier, cellText: String, cellTextColor: Color, fontWeight: FontWeight, + modifier: Modifier = Modifier, textAlpha: Float = 1f, ) { Text( diff --git a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/CompletionRatioProgressBar.kt b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/CompletionRatioProgressBar.kt similarity index 95% rename from feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/CompletionRatioProgressBar.kt rename to core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/CompletionRatioProgressBar.kt index 9746ff4e..e729c1b8 100644 --- a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/CompletionRatioProgressBar.kt +++ b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/CompletionRatioProgressBar.kt @@ -1,4 +1,4 @@ -package com.nexters.bandalart.android.feature.home.ui +package com.nexters.bandalart.android.core.ui.component import androidx.compose.animation.animateContentSize import androidx.compose.animation.core.LinearOutSlowInEasing @@ -29,6 +29,7 @@ import com.nexters.bandalart.android.core.designsystem.theme.Gray100 fun CompletionRatioProgressBar( completionRatio: Int, progressColor: Color, + modifier: Modifier = Modifier, ) { var progress by remember { mutableFloatStateOf(0f) } @@ -46,7 +47,7 @@ fun CompletionRatioProgressBar( } Column( - modifier = Modifier + modifier = modifier .fillMaxWidth() .wrapContentSize(), ) { diff --git a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/EmojiText.kt b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/EmojiText.kt index 0a86c955..fac366a0 100644 --- a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/EmojiText.kt +++ b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/EmojiText.kt @@ -2,16 +2,19 @@ package com.nexters.bandalart.android.core.ui.component import androidx.compose.material3.Text import androidx.compose.runtime.Composable +import androidx.compose.ui.Modifier import androidx.compose.ui.unit.TextUnit -import com.nexters.bandalart.android.core.ui.extension.nonScaleSp +import com.nexters.bandalart.android.core.ui.nonScaleSp @Composable fun EmojiText( emojiText: String?, fontSize: TextUnit, + modifier: Modifier = Modifier, ) { Text( text = emojiText ?: "", + modifier = modifier, fontSize = fontSize.nonScaleSp, ) } diff --git a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/FixedSizeText.kt b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/FixedSizeText.kt index ad2003c8..5dc5939d 100644 --- a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/FixedSizeText.kt +++ b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/FixedSizeText.kt @@ -8,16 +8,16 @@ import androidx.compose.ui.text.font.FontFamily import androidx.compose.ui.text.font.FontWeight import androidx.compose.ui.text.style.TextAlign import androidx.compose.ui.unit.TextUnit -import com.nexters.bandalart.android.core.ui.extension.nonScaleSp +import com.nexters.bandalart.android.core.ui.nonScaleSp import com.nexters.bandalart.android.core.designsystem.theme.pretendard @Composable fun FixedSizeText( text: String, - modifier: Modifier = Modifier, color: Color, fontSize: TextUnit, fontWeight: FontWeight, + modifier: Modifier = Modifier, fontFamily: FontFamily = pretendard, letterSpacing: TextUnit = TextUnit.Unspecified, textAlign: TextAlign? = null, diff --git a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/LoadingScreen.kt b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/LoadingScreen.kt index 77cbc00f..737ed4f3 100644 --- a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/LoadingScreen.kt +++ b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/LoadingScreen.kt @@ -23,7 +23,9 @@ import com.nexters.bandalart.android.core.designsystem.theme.Black import com.nexters.bandalart.android.core.designsystem.theme.White @Composable -fun LoadingScreen(modifier: Modifier = Modifier) { +fun LoadingScreen( + modifier: Modifier = Modifier, +) { Box( modifier = modifier.noRippleClickable { }, contentAlignment = Alignment.Center, diff --git a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/NetworkErrorAlertDialog.kt b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/NetworkErrorAlertDialog.kt index ab095196..827ea6f7 100644 --- a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/NetworkErrorAlertDialog.kt +++ b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/NetworkErrorAlertDialog.kt @@ -24,17 +24,17 @@ import androidx.compose.ui.unit.dp import androidx.compose.ui.unit.sp import androidx.compose.ui.window.Dialog import com.nexters.bandalart.android.core.ui.R -import com.nexters.bandalart.android.core.ui.extension.nonScaleSp +import com.nexters.bandalart.android.core.ui.nonScaleSp import com.nexters.bandalart.android.core.designsystem.theme.Gray400 import com.nexters.bandalart.android.core.designsystem.theme.Gray900 import com.nexters.bandalart.android.core.designsystem.theme.White @Composable fun NetworkErrorAlertDialog( - modifier: Modifier = Modifier, title: String, message: String, onConfirmClick: () -> Unit, + modifier: Modifier = Modifier, ) { val context = LocalContext.current Dialog(onDismissRequest = {}) { @@ -54,8 +54,8 @@ fun NetworkErrorAlertDialog( ) Spacer(modifier = Modifier.height(8.dp)) FixedSizeText( - modifier = Modifier.align(Alignment.CenterHorizontally), text = title, + modifier = Modifier.align(Alignment.CenterHorizontally), color = Gray900, fontSize = 20.sp, fontWeight = FontWeight.W700, @@ -65,8 +65,8 @@ fun NetworkErrorAlertDialog( ) Spacer(modifier = Modifier.height(8.dp)) FixedSizeText( - modifier = Modifier.align(Alignment.CenterHorizontally), text = message, + modifier = Modifier.align(Alignment.CenterHorizontally), color = Gray400, fontSize = 14.sp, fontWeight = FontWeight.W500, diff --git a/feature/onboarding/src/main/kotlin/com/nexters/bandalart/android/feature/onboarding/PagerIndicator.kt b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/PagerIndicator.kt similarity index 92% rename from feature/onboarding/src/main/kotlin/com/nexters/bandalart/android/feature/onboarding/PagerIndicator.kt rename to core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/PagerIndicator.kt index 416fede5..7a7c1df6 100644 --- a/feature/onboarding/src/main/kotlin/com/nexters/bandalart/android/feature/onboarding/PagerIndicator.kt +++ b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/PagerIndicator.kt @@ -1,6 +1,6 @@ @file:OptIn(ExperimentalFoundationApi::class) -package com.nexters.bandalart.android.feature.onboarding +package com.nexters.bandalart.android.core.ui.component import androidx.compose.foundation.ExperimentalFoundationApi import androidx.compose.foundation.background @@ -25,8 +25,9 @@ import com.nexters.bandalart.android.core.designsystem.theme.Gray700 fun PagerIndicator( pageCount: Int, pagerState: PagerState, + modifier: Modifier = Modifier, ) { - Box { + Box(modifier = modifier) { Row( modifier = Modifier .height(32.dp) diff --git a/feature/complete/src/main/kotlin/com/nexters/bandalart/android/feature/complete/ui/SaveImageButton.kt b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/SaveImageButton.kt similarity index 94% rename from feature/complete/src/main/kotlin/com/nexters/bandalart/android/feature/complete/ui/SaveImageButton.kt rename to core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/SaveImageButton.kt index d6e77e1f..f62b644e 100644 --- a/feature/complete/src/main/kotlin/com/nexters/bandalart/android/feature/complete/ui/SaveImageButton.kt +++ b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/SaveImageButton.kt @@ -1,6 +1,6 @@ @file:Suppress("unused") -package com.nexters.bandalart.android.feature.complete.ui +package com.nexters.bandalart.android.core.ui.component import androidx.compose.foundation.Image import androidx.compose.foundation.clickable @@ -21,7 +21,7 @@ import androidx.compose.ui.text.style.TextDecoration import androidx.compose.ui.text.withStyle import androidx.compose.ui.unit.dp import androidx.compose.ui.unit.sp -import com.nexters.bandalart.android.core.ui.extension.nonScaleSp +import com.nexters.bandalart.android.core.ui.nonScaleSp import com.nexters.bandalart.android.core.designsystem.theme.Gray400 import com.nexters.bandalart.android.core.designsystem.theme.pretendard import com.nexters.bandalart.android.core.designsystem.R diff --git a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/TitleText.kt b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/TitleText.kt index a1a5abf5..876d1f2e 100644 --- a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/TitleText.kt +++ b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/TitleText.kt @@ -2,18 +2,21 @@ package com.nexters.bandalart.android.core.ui.component import androidx.compose.material3.Text import androidx.compose.runtime.Composable +import androidx.compose.ui.Modifier import androidx.compose.ui.text.font.FontWeight import androidx.compose.ui.text.style.TextAlign import androidx.compose.ui.unit.sp -import com.nexters.bandalart.android.core.ui.extension.nonScaleSp +import com.nexters.bandalart.android.core.ui.nonScaleSp import com.nexters.bandalart.android.core.designsystem.theme.pretendard @Composable fun TitleText( text: String, + modifier: Modifier = Modifier, ) { Text( text = text, + modifier = modifier, fontFamily = pretendard, fontWeight = FontWeight.W700, fontSize = 22.sp.nonScaleSp, diff --git a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/bottomsheet/BottomSheetButton.kt b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/bottomsheet/BottomSheetButton.kt index d62fca87..174dbb66 100644 --- a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/bottomsheet/BottomSheetButton.kt +++ b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/bottomsheet/BottomSheetButton.kt @@ -15,8 +15,8 @@ import com.nexters.bandalart.android.core.ui.R @Composable fun BottomSheetDeleteButton( - modifier: Modifier = Modifier, onClick: () -> Unit, + modifier: Modifier = Modifier, ) { FilledIconButton( onClick = onClick, @@ -32,9 +32,9 @@ fun BottomSheetDeleteButton( @Composable fun BottomSheetCompleteButton( - modifier: Modifier = Modifier, isBlankCell: Boolean, onClick: () -> Unit, + modifier: Modifier = Modifier, ) { FilledIconButton( onClick = onClick, diff --git a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/bottomsheet/BottomSheetDivider.kt b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/bottomsheet/BottomSheetDivider.kt index 342faa94..8fdb6963 100644 --- a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/bottomsheet/BottomSheetDivider.kt +++ b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/bottomsheet/BottomSheetDivider.kt @@ -10,9 +10,11 @@ import androidx.compose.ui.unit.dp import com.nexters.bandalart.android.core.designsystem.theme.Gray300 @Composable -fun BottomSheetDivider() { +fun BottomSheetDivider( + modifier: Modifier = Modifier, +) { Box( - modifier = Modifier + modifier = modifier .height(1.dp) .fillMaxWidth() .background(Gray300), diff --git a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/bottomsheet/BottomSheetText.kt b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/bottomsheet/BottomSheetText.kt index 17f8749d..5f17a45f 100644 --- a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/bottomsheet/BottomSheetText.kt +++ b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/bottomsheet/BottomSheetText.kt @@ -16,10 +16,10 @@ import com.nexters.bandalart.android.core.ui.R @Composable fun BottomSheetTitleText( - modifier: Modifier = Modifier, isMainCell: Boolean, isSubCell: Boolean, isBlankCell: Boolean, + modifier: Modifier = Modifier, ) { FixedSizeText( text = diff --git a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/bottomsheet/BottomSheetTextStyle.kt b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/bottomsheet/BottomSheetTextStyle.kt index 220c0a3f..21c05950 100644 --- a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/bottomsheet/BottomSheetTextStyle.kt +++ b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/bottomsheet/BottomSheetTextStyle.kt @@ -4,7 +4,7 @@ import androidx.compose.runtime.Composable import androidx.compose.ui.text.TextStyle import androidx.compose.ui.text.font.FontWeight import androidx.compose.ui.unit.sp -import com.nexters.bandalart.android.core.ui.extension.nonScaleSp +import com.nexters.bandalart.android.core.ui.nonScaleSp import com.nexters.bandalart.android.core.designsystem.theme.Gray900 import com.nexters.bandalart.android.core.designsystem.theme.pretendard diff --git a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/bottomsheet/BottomSheetTopBar.kt b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/bottomsheet/BottomSheetTopBar.kt index 1b0a2154..91522319 100644 --- a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/bottomsheet/BottomSheetTopBar.kt +++ b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/component/bottomsheet/BottomSheetTopBar.kt @@ -31,11 +31,12 @@ fun BottomSheetTopBar( bottomSheetState: SheetState, onResult: (Boolean, Boolean) -> Unit, bottomSheetClosed: () -> Unit, + modifier: Modifier = Modifier, ) { val scope = rememberCoroutineScope() Box( - modifier = Modifier + modifier = modifier .fillMaxWidth() .padding(horizontal = 16.dp), ) { diff --git a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/extension/Modifier.kt b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/extension/Modifier.kt index 7ff8dfb9..38d8fe88 100644 --- a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/extension/Modifier.kt +++ b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/extension/Modifier.kt @@ -9,6 +9,8 @@ import androidx.compose.ui.Modifier import androidx.compose.ui.composed import androidx.compose.ui.platform.debugInspectorInfo import androidx.compose.ui.semantics.Role +import com.nexters.bandalart.android.core.ui.MultipleEventsCutter +import com.nexters.bandalart.android.core.ui.get @SuppressLint("ModifierFactoryUnreferencedReceiver") inline fun Modifier.noRippleClickable(crossinline onClick: () -> Unit): Modifier = composed { diff --git a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/extension/String.kt b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/extension/String.kt index 971717b6..f058b1f6 100644 --- a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/extension/String.kt +++ b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/extension/String.kt @@ -1,31 +1,7 @@ package com.nexters.bandalart.android.core.ui.extension import androidx.compose.ui.graphics.Color -import java.time.LocalDateTime -import java.time.format.DateTimeFormatter -import kotlinx.datetime.Instant -import kotlinx.datetime.TimeZone -import kotlinx.datetime.toLocalDateTime fun String.toColor(): Color { return Color(android.graphics.Color.parseColor(this)) } - -// ISO 8601 형식의 날짜와 시간 문자열을 "~년 월 일" 형태로 변환 -// "2023-08-02T18:27:25.862Z" -> "~23년 8월 2일" -fun String.toFormatDate(): String { - val instant = Instant.parse(this) - val dateTime = instant.toLocalDateTime(TimeZone.UTC) - return "~${dateTime.year - 2000}년 ${dateTime.monthNumber}월 ${dateTime.dayOfMonth}일" -} - -fun String.toStringLocalDateTime(): String { - val formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd'T'HH:mm") - val dueDate = LocalDateTime.parse(this.substring(0, 16), formatter) - return "${dueDate.year}년 ${dueDate.monthValue}월 ${dueDate.dayOfMonth}일" -} - -fun String.toLocalDateTime(): LocalDateTime { - val formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd'T'HH:mm") - return LocalDateTime.parse(this.substring(0, 16), formatter) -} diff --git a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/extension/UiText.kt b/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/extension/UiText.kt deleted file mode 100644 index c18b6393..00000000 --- a/core/ui/src/main/kotlin/com/nexters/bandalart/android/core/ui/extension/UiText.kt +++ /dev/null @@ -1,14 +0,0 @@ -package com.nexters.bandalart.android.core.ui.extension - -import android.content.Context -import androidx.annotation.StringRes - -sealed class UiText { - class StringResource(@StringRes val resId: Int) : UiText() - data class DirectString(val value: String) : UiText() - - fun asString(context: Context) = when (this) { - is StringResource -> context.getString(resId) - is DirectString -> value - } -} diff --git a/core/util/build.gradle.kts b/core/util/build.gradle.kts new file mode 100644 index 00000000..a11188c3 --- /dev/null +++ b/core/util/build.gradle.kts @@ -0,0 +1,20 @@ +@file:Suppress("UnstableApiUsage", "INLINE_FROM_HIGHER_PLATFORM") + +plugins { + bandalart("android-library") +} + +android { + namespace = "com.nexters.bandalart.android.core.util" + + buildFeatures { + buildConfig = true + } +} + +dependencies { + implementations( + libs.kotlinx.datetime, + libs.androidx.core, + ) +} diff --git a/core/util/src/main/kotlin/com/nexters/bandalart/android/core/util/extension/Context.kt b/core/util/src/main/kotlin/com/nexters/bandalart/android/core/util/extension/Context.kt new file mode 100644 index 00000000..b9c67fde --- /dev/null +++ b/core/util/src/main/kotlin/com/nexters/bandalart/android/core/util/extension/Context.kt @@ -0,0 +1,8 @@ +package com.nexters.bandalart.android.core.util.extension + +import android.content.Context +import java.util.Locale + +fun Context.getCurrentLocale(): Locale { + return this.resources.configuration.locales.get(0) +} diff --git a/core/util/src/main/kotlin/com/nexters/bandalart/android/core/util/extension/String.kt b/core/util/src/main/kotlin/com/nexters/bandalart/android/core/util/extension/String.kt new file mode 100644 index 00000000..229284cf --- /dev/null +++ b/core/util/src/main/kotlin/com/nexters/bandalart/android/core/util/extension/String.kt @@ -0,0 +1,26 @@ +package com.nexters.bandalart.android.core.util.extension + +import kotlinx.datetime.Instant +import kotlinx.datetime.TimeZone +import kotlinx.datetime.toLocalDateTime +import java.time.LocalDateTime +import java.time.format.DateTimeFormatter + +// ISO 8601 형식의 날짜와 시간 문자열을 "~년 월 일" 형태로 변환 +// "2023-08-02T18:27:25.862Z" -> "~23년 8월 2일" +fun String.toFormatDate(): String { + val instant = Instant.parse(this) + val dateTime = instant.toLocalDateTime(TimeZone.UTC) + return "~${dateTime.year - 2000}년 ${dateTime.monthNumber}월 ${dateTime.dayOfMonth}일" +} + +fun String.toStringLocalDateTime(): String { + val formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd'T'HH:mm") + val dueDate = LocalDateTime.parse(this.substring(0, 16), formatter) + return "${dueDate.year}년 ${dueDate.monthValue}월 ${dueDate.dayOfMonth}일" +} + +fun String.toLocalDateTime(): LocalDateTime { + val formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd'T'HH:mm") + return LocalDateTime.parse(this.substring(0, 16), formatter) +} diff --git a/feature/complete/src/main/kotlin/com/nexters/bandalart/android/feature/complete/CompleteScreen.kt b/feature/complete/src/main/kotlin/com/nexters/bandalart/android/feature/complete/CompleteScreen.kt index 4d0d2d52..29ef6ada 100644 --- a/feature/complete/src/main/kotlin/com/nexters/bandalart/android/feature/complete/CompleteScreen.kt +++ b/feature/complete/src/main/kotlin/com/nexters/bandalart/android/feature/complete/CompleteScreen.kt @@ -24,6 +24,7 @@ import com.airbnb.lottie.compose.LottieConstants import com.airbnb.lottie.compose.animateLottieCompositionAsState import com.airbnb.lottie.compose.rememberLottieComposition import com.nexters.bandalart.android.core.designsystem.theme.Gray50 +import com.nexters.bandalart.android.core.ui.ObserveAsEvents import com.nexters.bandalart.android.core.ui.R import com.nexters.bandalart.android.core.ui.component.BandalartButton import com.nexters.bandalart.android.core.ui.component.TitleText @@ -32,39 +33,41 @@ import com.nexters.bandalart.android.feature.complete.ui.CompleteTopBar @Composable internal fun CompleteRoute( - modifier: Modifier = Modifier, onNavigateBack: () -> Unit, + modifier: Modifier = Modifier, viewModel: CompleteViewModel = hiltViewModel(), ) { val context = LocalContext.current val uiState by viewModel.uiState.collectAsStateWithLifecycle() - LaunchedEffect(viewModel) { - viewModel.eventFlow.collect { event -> - when (event) { - is CompleteUiEvent.ShowToast -> { - Toast.makeText(context, event.message.asString(context), Toast.LENGTH_SHORT).show() - } + ObserveAsEvents(flow = viewModel.eventFlow) { event -> + when (event) { + is CompleteUiEvent.NavigateToHome -> { + onNavigateBack() + } + + is CompleteUiEvent.ShowToast -> { + Toast.makeText(context, event.message.asString(context), Toast.LENGTH_SHORT).show() } } } CompleteScreen( - modifier = modifier, uiState = uiState, - onNavigateBack = onNavigateBack, + navigateToHome = viewModel::navigateToHome, shareBandalart = viewModel::shareBandalart, initShareUrl = viewModel::initShareUrl, + modifier = modifier, ) } @Composable internal fun CompleteScreen( - modifier: Modifier = Modifier, uiState: CompleteUiState, - onNavigateBack: () -> Unit, + navigateToHome: () -> Unit, shareBandalart: () -> Unit, initShareUrl: () -> Unit, + modifier: Modifier = Modifier, ) { val context = LocalContext.current val composition by rememberLottieComposition( @@ -108,7 +111,7 @@ internal fun CompleteScreen( horizontalAlignment = Alignment.CenterHorizontally, ) { Spacer(modifier = Modifier.height(16.dp)) - CompleteTopBar(onNavigateBack = onNavigateBack) + CompleteTopBar(onNavigateBack = navigateToHome) Spacer(modifier = Modifier.height(40.dp)) TitleText(text = context.getString(R.string.complete_title)) Box(modifier = Modifier.fillMaxSize()) { diff --git a/feature/complete/src/main/kotlin/com/nexters/bandalart/android/feature/complete/CompleteViewModel.kt b/feature/complete/src/main/kotlin/com/nexters/bandalart/android/feature/complete/CompleteViewModel.kt index ffa0c50e..da140ec2 100644 --- a/feature/complete/src/main/kotlin/com/nexters/bandalart/android/feature/complete/CompleteViewModel.kt +++ b/feature/complete/src/main/kotlin/com/nexters/bandalart/android/feature/complete/CompleteViewModel.kt @@ -5,7 +5,7 @@ import androidx.lifecycle.ViewModel import androidx.lifecycle.viewModelScope import com.nexters.bandalart.android.core.domain.usecase.bandalart.ShareBandalartUseCase import com.nexters.bandalart.android.core.domain.usecase.bandalart.UpsertBandalartKeyUseCase -import com.nexters.bandalart.android.core.ui.extension.UiText +import com.nexters.bandalart.android.core.ui.UiText import com.nexters.bandalart.android.feature.complete.navigation.BANDALART_KEY import com.nexters.bandalart.android.feature.complete.navigation.BANDALART_PROFILE_EMOJI import com.nexters.bandalart.android.feature.complete.navigation.BANDALART_TITLE @@ -39,8 +39,9 @@ data class CompleteUiState( val error: Throwable? = null, ) -sealed class CompleteUiEvent { - data class ShowToast(val message: UiText) : CompleteUiEvent() +sealed interface CompleteUiEvent { + data object NavigateToHome : CompleteUiEvent + data class ShowToast(val message: UiText) : CompleteUiEvent } @HiltViewModel @@ -111,4 +112,10 @@ class CompleteViewModel @Inject constructor( fun initShareUrl() { _uiState.value = _uiState.value.copy(shareUrl = "") } + + fun navigateToHome() { + viewModelScope.launch { + _eventFlow.emit(CompleteUiEvent.NavigateToHome) + } + } } diff --git a/feature/complete/src/main/kotlin/com/nexters/bandalart/android/feature/complete/ui/CompleteBandalart.kt b/feature/complete/src/main/kotlin/com/nexters/bandalart/android/feature/complete/ui/CompleteBandalart.kt index 8514bab5..5d3df636 100644 --- a/feature/complete/src/main/kotlin/com/nexters/bandalart/android/feature/complete/ui/CompleteBandalart.kt +++ b/feature/complete/src/main/kotlin/com/nexters/bandalart/android/feature/complete/ui/CompleteBandalart.kt @@ -34,8 +34,8 @@ import com.nexters.bandalart.android.feature.complete.CompleteUiState @Composable fun CompleteBandalart( - modifier: Modifier = Modifier, uiState: CompleteUiState, + modifier: Modifier = Modifier, ) { val context = LocalContext.current diff --git a/feature/complete/src/main/kotlin/com/nexters/bandalart/android/feature/complete/ui/CompleteTopBar.kt b/feature/complete/src/main/kotlin/com/nexters/bandalart/android/feature/complete/ui/CompleteTopBar.kt index 6940dea1..b2b0b6d0 100644 --- a/feature/complete/src/main/kotlin/com/nexters/bandalart/android/feature/complete/ui/CompleteTopBar.kt +++ b/feature/complete/src/main/kotlin/com/nexters/bandalart/android/feature/complete/ui/CompleteTopBar.kt @@ -6,7 +6,7 @@ import androidx.compose.foundation.layout.fillMaxWidth import androidx.compose.foundation.layout.padding import androidx.compose.foundation.layout.width import androidx.compose.material.icons.Icons -import androidx.compose.material.icons.filled.ArrowBackIos +import androidx.compose.material.icons.automirrored.filled.ArrowBackIos import androidx.compose.material3.Icon import androidx.compose.material3.IconButton import androidx.compose.runtime.Composable @@ -18,12 +18,12 @@ import com.nexters.bandalart.android.core.ui.R @Composable fun CompleteTopBar( - modifier: Modifier = Modifier, onNavigateBack: () -> Unit, + modifier: Modifier = Modifier, ) { val context = LocalContext.current Row( - modifier + modifier = modifier .fillMaxWidth() .padding(horizontal = 16.dp), ) { @@ -34,7 +34,7 @@ fun CompleteTopBar( .aspectRatio(1f), ) { Icon( - imageVector = Icons.Default.ArrowBackIos, + imageVector = Icons.AutoMirrored.Filled.ArrowBackIos, contentDescription = context.getString(R.string.arrow_forward_descrption), tint = Gray900, ) diff --git a/feature/home/build.gradle.kts b/feature/home/build.gradle.kts index 5a64bc08..a188ec05 100644 --- a/feature/home/build.gradle.kts +++ b/feature/home/build.gradle.kts @@ -19,6 +19,7 @@ dependencies { projects.core.designsystem, projects.core.domain, projects.core.ui, + projects.core.util, libs.kotlinx.collections.immutable, libs.androidx.core, libs.androidx.hilt.compose.navigation, diff --git a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/BandalartBottomSheet.kt b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/BandalartBottomSheet.kt index b5df9bbc..34ad38c4 100644 --- a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/BandalartBottomSheet.kt +++ b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/BandalartBottomSheet.kt @@ -7,7 +7,6 @@ import androidx.compose.animation.AnimatedVisibility import androidx.compose.animation.animateContentSize import androidx.compose.animation.core.LinearOutSlowInEasing import androidx.compose.animation.core.tween -import com.nexters.bandalart.android.core.ui.R import androidx.compose.foundation.Image import androidx.compose.foundation.background import androidx.compose.foundation.clickable @@ -33,7 +32,7 @@ import androidx.compose.foundation.text.KeyboardActions import androidx.compose.foundation.text.KeyboardOptions import androidx.compose.foundation.verticalScroll import androidx.compose.material.icons.Icons -import androidx.compose.material.icons.filled.ArrowForwardIos +import androidx.compose.material.icons.automirrored.filled.ArrowForwardIos import androidx.compose.material3.Card import androidx.compose.material3.CardDefaults import androidx.compose.material3.ExperimentalMaterial3Api @@ -62,6 +61,16 @@ import androidx.compose.ui.unit.dp import androidx.compose.ui.unit.sp import androidx.hilt.navigation.compose.hiltViewModel import androidx.lifecycle.compose.collectAsStateWithLifecycle +import com.nexters.bandalart.android.core.designsystem.theme.Gray100 +import com.nexters.bandalart.android.core.designsystem.theme.Gray300 +import com.nexters.bandalart.android.core.designsystem.theme.Gray400 +import com.nexters.bandalart.android.core.designsystem.theme.Gray700 +import com.nexters.bandalart.android.core.designsystem.theme.Transparent +import com.nexters.bandalart.android.core.designsystem.theme.White +import com.nexters.bandalart.android.core.ui.R +import com.nexters.bandalart.android.core.ui.StatusBarHeightDp +import com.nexters.bandalart.android.core.ui.ThemeColor +import com.nexters.bandalart.android.core.ui.allColor import com.nexters.bandalart.android.core.ui.component.BandalartDeleteAlertDialog import com.nexters.bandalart.android.core.ui.component.EmojiText import com.nexters.bandalart.android.core.ui.component.bottomsheet.BottomSheetCompleteButton @@ -72,20 +81,11 @@ import com.nexters.bandalart.android.core.ui.component.bottomsheet.BottomSheetDi import com.nexters.bandalart.android.core.ui.component.bottomsheet.BottomSheetSubTitleText import com.nexters.bandalart.android.core.ui.component.bottomsheet.BottomSheetTextStyle import com.nexters.bandalart.android.core.ui.component.bottomsheet.BottomSheetTopBar -import com.nexters.bandalart.android.core.ui.extension.NavigationBarHeightDp -import com.nexters.bandalart.android.core.ui.extension.StatusBarHeightDp -import com.nexters.bandalart.android.core.ui.extension.ThemeColor -import com.nexters.bandalart.android.core.ui.extension.allColor import com.nexters.bandalart.android.core.ui.extension.noRippleClickable -import com.nexters.bandalart.android.core.ui.extension.nonScaleSp -import com.nexters.bandalart.android.core.ui.extension.toLocalDateTime -import com.nexters.bandalart.android.core.ui.extension.toStringLocalDateTime -import com.nexters.bandalart.android.core.designsystem.theme.Gray100 -import com.nexters.bandalart.android.core.designsystem.theme.Gray300 -import com.nexters.bandalart.android.core.designsystem.theme.Gray400 -import com.nexters.bandalart.android.core.designsystem.theme.Gray700 -import com.nexters.bandalart.android.core.designsystem.theme.Transparent -import com.nexters.bandalart.android.core.designsystem.theme.White +import com.nexters.bandalart.android.core.ui.getNavigationBarPadding +import com.nexters.bandalart.android.core.ui.nonScaleSp +import com.nexters.bandalart.android.core.util.extension.toLocalDateTime +import com.nexters.bandalart.android.core.util.extension.toStringLocalDateTime import com.nexters.bandalart.android.feature.home.model.BandalartCellUiModel import com.nexters.bandalart.android.feature.home.model.UpdateBandalartMainCellModel import com.nexters.bandalart.android.feature.home.model.UpdateBandalartSubCellModel @@ -93,8 +93,8 @@ import com.nexters.bandalart.android.feature.home.model.UpdateBandalartTaskCellM import com.nexters.bandalart.android.feature.home.ui.bandalart.BandalartColorPicker import com.nexters.bandalart.android.feature.home.ui.bandalart.BandalartDatePicker import com.nexters.bandalart.android.feature.home.ui.bandalart.BandalartEmojiPicker -import java.time.LocalDateTime import kotlinx.coroutines.launch +import java.time.LocalDateTime @Composable fun BandalartBottomSheet( @@ -107,6 +107,7 @@ fun BandalartBottomSheet( bottomSheetState: Boolean, bottomSheetDataChangedState: Boolean, ) -> Unit, + modifier: Modifier = Modifier, viewModel: BottomSheetViewModel = hiltViewModel(), ) { val uiState by viewModel.uiState.collectAsStateWithLifecycle() @@ -122,7 +123,7 @@ fun BandalartBottomSheet( viewModel.bottomSheetClosed() onResult(false, false) }, - modifier = Modifier + modifier = modifier .wrapContentSize() .statusBarsPadding() .noRippleClickable { }, @@ -209,7 +210,6 @@ fun BandalartBottomSheet( onResult = onResult, bottomSheetClosed = viewModel::bottomSheetClosed, ) - Box { Column( modifier = Modifier @@ -363,7 +363,7 @@ fun BandalartBottomSheet( .align(Alignment.CenterEnd) .height(21.dp) .aspectRatio(1f), - imageVector = Icons.Default.ArrowForwardIos, + imageVector = Icons.AutoMirrored.Filled.ArrowForwardIos, contentDescription = stringResource(R.string.arrow_forward_descrption), tint = Gray400, ) @@ -499,7 +499,7 @@ fun BandalartBottomSheet( }, ) } - Spacer(modifier = Modifier.height(StatusBarHeightDp + NavigationBarHeightDp + 20.dp)) + Spacer(modifier = Modifier.height(StatusBarHeightDp + getNavigationBarPadding())) } if (scrollState.value > 0) { Column( diff --git a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/BandalartBottomSheetViewModel.kt b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/BandalartBottomSheetViewModel.kt index 9eb71e7a..5b249706 100644 --- a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/BandalartBottomSheetViewModel.kt +++ b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/BandalartBottomSheetViewModel.kt @@ -6,12 +6,12 @@ import com.nexters.bandalart.android.core.domain.usecase.bandalart.DeleteBandala import com.nexters.bandalart.android.core.domain.usecase.bandalart.UpdateBandalartMainCellUseCase import com.nexters.bandalart.android.core.domain.usecase.bandalart.UpdateBandalartSubCellUseCase import com.nexters.bandalart.android.core.domain.usecase.bandalart.UpdateBandalartTaskCellUseCase +import com.nexters.bandalart.android.core.ui.UiText import com.nexters.bandalart.android.feature.home.mapper.toEntity import com.nexters.bandalart.android.feature.home.model.BandalartCellUiModel import com.nexters.bandalart.android.feature.home.model.UpdateBandalartMainCellModel import com.nexters.bandalart.android.feature.home.model.UpdateBandalartSubCellModel import com.nexters.bandalart.android.feature.home.model.UpdateBandalartTaskCellModel -import com.nexters.bandalart.android.core.ui.extension.UiText import dagger.hilt.android.lifecycle.HiltViewModel import javax.inject.Inject import kotlinx.coroutines.flow.MutableSharedFlow diff --git a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/HomeScreen.kt b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/HomeScreen.kt index ab776d86..3efb7b72 100644 --- a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/HomeScreen.kt +++ b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/HomeScreen.kt @@ -21,6 +21,7 @@ import androidx.compose.runtime.LaunchedEffect import androidx.compose.runtime.derivedStateOf import androidx.compose.runtime.getValue import androidx.compose.runtime.remember +import androidx.compose.runtime.rememberCoroutineScope import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.platform.LocalContext @@ -30,19 +31,20 @@ import androidx.hilt.navigation.compose.hiltViewModel import androidx.lifecycle.compose.collectAsStateWithLifecycle import com.nexters.bandalart.android.core.designsystem.theme.Gray100 import com.nexters.bandalart.android.core.designsystem.theme.Gray50 +import com.nexters.bandalart.android.core.ui.ObserveAsEvents import com.nexters.bandalart.android.core.ui.R +import com.nexters.bandalart.android.core.ui.ThemeColor import com.nexters.bandalart.android.core.ui.component.BandalartDeleteAlertDialog import com.nexters.bandalart.android.core.ui.component.LoadingScreen import com.nexters.bandalart.android.core.ui.component.NetworkErrorAlertDialog -import com.nexters.bandalart.android.core.ui.extension.ThemeColor import com.nexters.bandalart.android.feature.home.model.BandalartDetailUiModel +import com.nexters.bandalart.android.feature.home.ui.HomeHeader +import com.nexters.bandalart.android.feature.home.ui.HomeTopBar +import com.nexters.bandalart.android.feature.home.ui.ShareButton import com.nexters.bandalart.android.feature.home.ui.bandalart.BandalartChart import com.nexters.bandalart.android.feature.home.ui.bandalart.BandalartEmojiBottomSheet import com.nexters.bandalart.android.feature.home.ui.bandalart.BandalartListBottomSheet import com.nexters.bandalart.android.feature.home.ui.bandalart.BandalartSkeleton -import com.nexters.bandalart.android.feature.home.ui.HomeHeader -import com.nexters.bandalart.android.feature.home.ui.HomeTopBar -import com.nexters.bandalart.android.feature.home.ui.ShareButton import kotlinx.collections.immutable.toImmutableList import kotlinx.coroutines.delay import kotlinx.coroutines.launch @@ -51,40 +53,50 @@ private const val SnackbarDuration = 1000L @Composable internal fun HomeRoute( - modifier: Modifier = Modifier, navigateToComplete: (String, String, String) -> Unit, onShowSnackbar: suspend (String) -> Boolean, + modifier: Modifier = Modifier, viewModel: HomeViewModel = hiltViewModel(), ) { val uiState by viewModel.uiState.collectAsStateWithLifecycle() val context = LocalContext.current + val scope = rememberCoroutineScope() val bandalartCount by remember { derivedStateOf { uiState.bandalartList.size } } - LaunchedEffect(viewModel) { - viewModel.eventFlow.collect { event -> - when (event) { - is HomeUiEvent.ShowSnackbar -> { + ObserveAsEvents(flow = viewModel.eventFlow) { event -> + when (event) { + is HomeUiEvent.NavigateToComplete -> { + navigateToComplete( + event.key, + event.title, + event.profileEmoji.ifEmpty { + context.getString(R.string.home_default_emoji) + }, + ) + } + + is HomeUiEvent.ShowSnackbar -> { + scope.launch { val job = launch { onShowSnackbar(event.message.asString(context)) } delay(SnackbarDuration) job.cancel() } + } - is HomeUiEvent.ShowToast -> { - Toast.makeText(context, event.message.asString(context), Toast.LENGTH_SHORT).show() - } + is HomeUiEvent.ShowToast -> { + Toast.makeText(context, event.message.asString(context), Toast.LENGTH_SHORT).show() } } } HomeScreen( - modifier = modifier, uiState = uiState, bandalartCount = bandalartCount, - navigateToComplete = navigateToComplete, + navigateToComplete = viewModel::navigateToComplete, getBandalartList = viewModel::getBandalartList, getBandalartDetail = viewModel::getBandalartDetail, createBandalart = viewModel::createBandalart, @@ -102,15 +114,15 @@ internal fun HomeRoute( initShareUrl = viewModel::initShareUrl, checkCompletedBandalartKey = viewModel::checkCompletedBandalartKey, openNetworkErrorDialog = viewModel::openNetworkErrorAlertDialog, + modifier = modifier, ) } @Composable internal fun HomeScreen( - modifier: Modifier = Modifier, uiState: HomeUiState, bandalartCount: Int, - navigateToComplete: (String, String, String) -> Unit, + navigateToComplete: () -> Unit, getBandalartList: (String?) -> Unit, getBandalartDetail: (String) -> Unit, createBandalart: () -> Unit, @@ -128,6 +140,7 @@ internal fun HomeScreen( initShareUrl: () -> Unit, checkCompletedBandalartKey: suspend (String) -> Boolean, openNetworkErrorDialog: (Boolean) -> Unit, + modifier: Modifier = Modifier, ) { val context = LocalContext.current @@ -136,18 +149,13 @@ internal fun HomeScreen( } LaunchedEffect(key1 = uiState.bandalartDetailData?.isCompleted) { + val bandalartDetailData = uiState.bandalartDetailData ?: return@LaunchedEffect // 목표를 달성했을 경우 - if (uiState.bandalartDetailData?.isCompleted == true && !uiState.bandalartDetailData.title.isNullOrEmpty()) { + if (bandalartDetailData.isCompleted && !bandalartDetailData.title.isNullOrEmpty()) { // 목표 달성 화면을 띄워 줘야 하는 반다라트일 경우 - val isBandalartCompleted = checkCompletedBandalartKey(uiState.bandalartDetailData.key) + val isBandalartCompleted = checkCompletedBandalartKey(bandalartDetailData.key) if (isBandalartCompleted) { - navigateToComplete( - uiState.bandalartDetailData.key, - uiState.bandalartDetailData.title, - if (uiState.bandalartDetailData.profileEmoji.isNullOrEmpty()) { - context.getString(R.string.home_default_emoji) - } else uiState.bandalartDetailData.profileEmoji, - ) + navigateToComplete() } } } diff --git a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/HomeViewModel.kt b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/HomeViewModel.kt index 58bd95fb..d4feca2e 100644 --- a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/HomeViewModel.kt +++ b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/HomeViewModel.kt @@ -16,7 +16,7 @@ import com.nexters.bandalart.android.core.domain.usecase.bandalart.ShareBandalar import com.nexters.bandalart.android.core.domain.usecase.bandalart.UpdateBandalartEmojiUseCase import com.nexters.bandalart.android.core.domain.usecase.bandalart.UpsertBandalartKeyUseCase import com.nexters.bandalart.android.core.ui.R -import com.nexters.bandalart.android.core.ui.extension.UiText +import com.nexters.bandalart.android.core.ui.UiText import com.nexters.bandalart.android.feature.home.mapper.toEntity import com.nexters.bandalart.android.feature.home.mapper.toUiModel import com.nexters.bandalart.android.feature.home.model.BandalartCellUiModel @@ -82,9 +82,15 @@ data class HomeUiState( val error: Throwable? = null, ) -sealed class HomeUiEvent { - data class ShowSnackbar(val message: UiText) : HomeUiEvent() - data class ShowToast(val message: UiText) : HomeUiEvent() +sealed interface HomeUiEvent { + data class NavigateToComplete( + val key: String, + val title: String, + val profileEmoji: String, + ) : HomeUiEvent + + data class ShowSnackbar(val message: UiText) : HomeUiEvent + data class ShowToast(val message: UiText) : HomeUiEvent } @HiltViewModel @@ -501,4 +507,16 @@ class HomeViewModel @Inject constructor( deleteBandalartKeyUseCase(bandalartKey) } } + + fun navigateToComplete() { + viewModelScope.launch { + _eventFlow.emit( + HomeUiEvent.NavigateToComplete( + key = uiState.value.bandalartDetailData!!.key, + title = uiState.value.bandalartDetailData!!.title!!, + profileEmoji = uiState.value.bandalartDetailData!!.profileEmoji ?: "", + ), + ) + } + } } diff --git a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/HomeAppTitle.kt b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/HomeAppTitle.kt index fc12b97c..9a3d760d 100644 --- a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/HomeAppTitle.kt +++ b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/HomeAppTitle.kt @@ -1,6 +1,5 @@ package com.nexters.bandalart.android.feature.home.ui -import android.content.Context import androidx.compose.runtime.Composable import androidx.compose.ui.Modifier import androidx.compose.ui.platform.LocalContext @@ -12,6 +11,7 @@ import com.nexters.bandalart.android.core.designsystem.theme.koronaOneRegular import com.nexters.bandalart.android.core.designsystem.theme.neurimboGothicRegular import com.nexters.bandalart.android.core.ui.R import com.nexters.bandalart.android.core.ui.component.FixedSizeText +import com.nexters.bandalart.android.core.util.extension.getCurrentLocale import java.util.Locale @Composable @@ -41,8 +41,8 @@ fun HomeAppKoreanTitle( modifier: Modifier = Modifier, ) { FixedSizeText( - modifier = modifier, text = stringResource(R.string.bandalart), + modifier = modifier, color = Gray900, fontSize = 28.sp, fontWeight = FontWeight.W400, @@ -67,8 +67,3 @@ fun HomeAppEnglishTitle( letterSpacing = (-0.36).sp, ) } - -// TODO core:util 모듈로 옮길 예정 -fun Context.getCurrentLocale(): Locale { - return this.resources.configuration.locales.get(0) -} diff --git a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/HomeHeader.kt b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/HomeHeader.kt index f9de95d1..38a5517f 100644 --- a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/HomeHeader.kt +++ b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/HomeHeader.kt @@ -34,27 +34,26 @@ import androidx.compose.ui.unit.sp import com.nexters.bandalart.android.core.ui.component.BandalartDropDownMenu import com.nexters.bandalart.android.core.ui.component.EmojiText import com.nexters.bandalart.android.core.ui.component.FixedSizeText -import com.nexters.bandalart.android.core.ui.extension.toColor -import com.nexters.bandalart.android.core.ui.extension.toFormatDate import com.nexters.bandalart.android.core.designsystem.theme.Gray100 import com.nexters.bandalart.android.core.designsystem.theme.Gray300 import com.nexters.bandalart.android.core.designsystem.theme.Gray600 import com.nexters.bandalart.android.core.designsystem.theme.Gray900 import com.nexters.bandalart.android.core.designsystem.theme.MainColor +import com.nexters.bandalart.android.core.ui.component.CompletionRatioProgressBar +import com.nexters.bandalart.android.core.ui.extension.toColor +import com.nexters.bandalart.android.core.util.extension.toFormatDate import com.nexters.bandalart.android.feature.home.HomeUiState @Composable fun HomeHeader( - modifier: Modifier = Modifier, uiState: HomeUiState, openDropDownMenu: (Boolean) -> Unit, openEmojiBottomSheet: (Boolean) -> Unit, openBandalartDeleteAlertDialog: (Boolean) -> Unit, openCellBottomSheet: (Boolean) -> Unit, + modifier: Modifier = Modifier, ) { - Column( - modifier.padding(horizontal = 16.dp), - ) { + Column(modifier.padding(horizontal = 16.dp)) { Spacer(modifier = Modifier.height(24.dp)) Column { Box(modifier = Modifier.align(Alignment.CenterHorizontally)) { diff --git a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/ShareButton.kt b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/HomeShareButton.kt similarity index 97% rename from feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/ShareButton.kt rename to feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/HomeShareButton.kt index f1ea9f9c..d2bece33 100644 --- a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/ShareButton.kt +++ b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/HomeShareButton.kt @@ -19,16 +19,16 @@ import androidx.compose.ui.unit.sp import com.nexters.bandalart.android.core.designsystem.R import com.nexters.bandalart.android.core.ui.component.FixedSizeText import com.nexters.bandalart.android.core.ui.extension.clickableSingle -import com.nexters.bandalart.android.core.ui.extension.nonScaleSp +import com.nexters.bandalart.android.core.ui.nonScaleSp import com.nexters.bandalart.android.core.designsystem.theme.Gray100 import com.nexters.bandalart.android.core.designsystem.theme.Gray900 import com.nexters.bandalart.android.feature.home.HomeUiState @Composable fun ShareButton( - modifier: Modifier = Modifier, uiState: HomeUiState, shareBandalart: (String) -> Unit, + modifier: Modifier = Modifier, ) { Box( modifier = modifier diff --git a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/HomeTopBar.kt b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/HomeTopBar.kt index a12ef9fd..f6adc357 100644 --- a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/HomeTopBar.kt +++ b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/HomeTopBar.kt @@ -27,15 +27,16 @@ import com.nexters.bandalart.android.core.designsystem.theme.White import com.nexters.bandalart.android.core.designsystem.theme.pretendard import com.nexters.bandalart.android.core.ui.R import com.nexters.bandalart.android.core.ui.component.FixedSizeText -import com.nexters.bandalart.android.core.ui.extension.nonScaleSp +import com.nexters.bandalart.android.core.ui.nonScaleSp @Composable internal fun HomeTopBar( bandalartCount: Int, onShowBandalartList: () -> Unit, + modifier: Modifier = Modifier, ) { Box( - modifier = Modifier + modifier = modifier .fillMaxWidth() .height(62.dp) .background(White), diff --git a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartCell.kt b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartCell.kt index bb23ed29..7dd65aa6 100644 --- a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartCell.kt +++ b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartCell.kt @@ -30,13 +30,13 @@ import androidx.compose.ui.unit.Dp import androidx.compose.ui.unit.dp import com.nexters.bandalart.android.core.ui.R import com.nexters.bandalart.android.core.ui.component.CellText -import com.nexters.bandalart.android.core.ui.extension.ThemeColor -import com.nexters.bandalart.android.core.ui.extension.toColor import com.nexters.bandalart.android.core.designsystem.theme.Gray200 import com.nexters.bandalart.android.core.designsystem.theme.Gray400 import com.nexters.bandalart.android.core.designsystem.theme.Gray500 import com.nexters.bandalart.android.core.designsystem.theme.Gray900 import com.nexters.bandalart.android.core.designsystem.theme.White +import com.nexters.bandalart.android.core.ui.ThemeColor +import com.nexters.bandalart.android.core.ui.extension.toColor import com.nexters.bandalart.android.feature.home.BandalartBottomSheet import com.nexters.bandalart.android.feature.home.model.BandalartCellUiModel @@ -58,13 +58,13 @@ data class SubCell( @Composable fun BandalartCell( - modifier: Modifier = Modifier, bandalartKey: String, themeColor: ThemeColor, isMainCell: Boolean, - cellInfo: CellInfo = CellInfo(), cellData: BandalartCellUiModel, bottomSheetDataChanged: (Boolean) -> Unit, + modifier: Modifier = Modifier, + cellInfo: CellInfo = CellInfo(), outerPadding: Dp = 3.dp, innerPadding: Dp = 2.dp, mainCellPadding: Dp = 1.dp, diff --git a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartCellGrid.kt b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartCellGrid.kt index de23dd35..9d17e554 100644 --- a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartCellGrid.kt +++ b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartCellGrid.kt @@ -8,7 +8,7 @@ import androidx.compose.foundation.layout.fillMaxWidth import androidx.compose.runtime.Composable import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier -import com.nexters.bandalart.android.core.ui.extension.ThemeColor +import com.nexters.bandalart.android.core.ui.ThemeColor @Composable fun BandalartCellGrid( @@ -18,9 +18,10 @@ fun BandalartCellGrid( rows: Int, cols: Int, bottomSheetDataChanged: (Boolean) -> Unit, + modifier: Modifier = Modifier, ) { Column( - modifier = Modifier.fillMaxSize(), + modifier = modifier.fillMaxSize(), horizontalAlignment = Alignment.CenterHorizontally, verticalArrangement = Arrangement.SpaceEvenly, ) { diff --git a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartChart.kt b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartChart.kt index caf4d777..229711e3 100644 --- a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartChart.kt +++ b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartChart.kt @@ -16,19 +16,19 @@ import androidx.compose.ui.res.stringResource import androidx.compose.ui.unit.Constraints import androidx.compose.ui.unit.dp import com.nexters.bandalart.android.core.ui.R -import com.nexters.bandalart.android.core.ui.extension.ThemeColor -import com.nexters.bandalart.android.core.ui.extension.toColor import com.nexters.bandalart.android.core.designsystem.theme.Gray100 import com.nexters.bandalart.android.core.designsystem.theme.MainColor +import com.nexters.bandalart.android.core.ui.ThemeColor +import com.nexters.bandalart.android.core.ui.extension.toColor import com.nexters.bandalart.android.feature.home.HomeUiState @Composable fun BandalartChart( - modifier: Modifier = Modifier, bandalartKey: String, uiState: HomeUiState, themeColor: ThemeColor, bottomSheetDataChanged: (Boolean) -> Unit, + modifier: Modifier = Modifier, ) { val context = LocalContext.current val screenWidthDp = LocalConfiguration.current.screenWidthDp.dp diff --git a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartColorPicker.kt b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartColorPicker.kt index f0a6aa49..edaa1d5c 100644 --- a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartColorPicker.kt +++ b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartColorPicker.kt @@ -18,20 +18,21 @@ import androidx.compose.runtime.setValue import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.unit.dp -import com.nexters.bandalart.android.core.ui.extension.ThemeColor -import com.nexters.bandalart.android.core.ui.extension.allColor import com.nexters.bandalart.android.core.ui.extension.noRippleClickable -import com.nexters.bandalart.android.core.ui.extension.toColor import com.nexters.bandalart.android.core.designsystem.theme.Gray900 import com.nexters.bandalart.android.core.designsystem.theme.White +import com.nexters.bandalart.android.core.ui.ThemeColor +import com.nexters.bandalart.android.core.ui.allColor +import com.nexters.bandalart.android.core.ui.extension.toColor @Composable fun BandalartColorPicker( initColor: ThemeColor, onResult: (ThemeColor) -> Unit, + modifier: Modifier = Modifier, ) { Row( - modifier = Modifier + modifier = modifier .fillMaxWidth() .height(45.dp), verticalAlignment = Alignment.CenterVertically, diff --git a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartDatePicker.kt b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartDatePicker.kt index 77fecc19..22a1c33a 100644 --- a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartDatePicker.kt +++ b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartDatePicker.kt @@ -34,11 +34,11 @@ import androidx.compose.ui.text.style.TextAlign import androidx.compose.ui.unit.dp import androidx.compose.ui.unit.sp import com.nexters.bandalart.android.core.ui.R -import com.nexters.bandalart.android.core.ui.extension.nonScaleSp -import com.nexters.bandalart.android.core.ui.extension.toLocalDateTime +import com.nexters.bandalart.android.core.ui.nonScaleSp import com.nexters.bandalart.android.core.designsystem.theme.Gray100 import com.nexters.bandalart.android.core.designsystem.theme.Gray900 import com.nexters.bandalart.android.core.designsystem.theme.pretendard +import com.nexters.bandalart.android.core.util.extension.toLocalDateTime import kotlinx.collections.immutable.ImmutableList import kotlinx.collections.immutable.toImmutableList import java.time.LocalDateTime @@ -52,10 +52,11 @@ fun BandalartDatePicker( datePickerScope: CoroutineScope, datePickerState: SheetState, currentDueDate: LocalDateTime, + modifier: Modifier = Modifier, ) { Column( + modifier = modifier.fillMaxWidth(), horizontalAlignment = Alignment.CenterHorizontally, - modifier = Modifier.fillMaxWidth(), ) { val chosenYear = remember { mutableStateOf(currentDueDate.year.toString()) } val chosenMonth = remember { mutableStateOf(currentDueDate.monthValue.toString()) } diff --git a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartEmojiBottomSheet.kt b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartEmojiBottomSheet.kt index c16f615a..586653c7 100644 --- a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartEmojiBottomSheet.kt +++ b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartEmojiBottomSheet.kt @@ -22,6 +22,7 @@ fun BandalartEmojiBottomSheet( bottomSheetState: Boolean, bottomSheetDataChangedState: Boolean, ) -> Unit, + modifier: Modifier = Modifier, viewModel: HomeViewModel = hiltViewModel(), ) { val bottomSheetState = rememberModalBottomSheetState(skipPartiallyExpanded = true) @@ -30,8 +31,7 @@ fun BandalartEmojiBottomSheet( onDismissRequest = { onResult(false, false) }, - modifier = Modifier - .wrapContentSize(), + modifier = modifier.wrapContentSize(), sheetState = bottomSheetState, dragHandle = null, ) { diff --git a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartEmojiPicker.kt b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartEmojiPicker.kt index d55d7e30..b74a9a77 100644 --- a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartEmojiPicker.kt +++ b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartEmojiPicker.kt @@ -34,17 +34,17 @@ import com.nexters.bandalart.android.core.designsystem.theme.Gray100 import com.nexters.bandalart.android.core.designsystem.theme.Gray400 import com.nexters.bandalart.android.core.designsystem.theme.White import com.nexters.bandalart.android.core.ui.component.EmojiText -import com.nexters.bandalart.android.core.ui.extension.NavigationBarHeightDp +import com.nexters.bandalart.android.core.ui.NavigationBarHeightDp import java.util.* import kotlinx.coroutines.launch @Composable fun BandalartEmojiPicker( - modifier: Modifier = Modifier, currentEmoji: String?, isBottomSheet: Boolean, onResult: (String?, Boolean) -> Unit, emojiPickerState: SheetState, + modifier: Modifier = Modifier, ): @Composable (ColumnScope.() -> Unit) { return { val scope = rememberCoroutineScope() diff --git a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartItem.kt b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartItem.kt index 07bcdd39..21411bed 100644 --- a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartItem.kt +++ b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartItem.kt @@ -16,7 +16,7 @@ import androidx.compose.foundation.layout.size import androidx.compose.foundation.layout.width import androidx.compose.foundation.shape.RoundedCornerShape import androidx.compose.material.icons.Icons -import androidx.compose.material.icons.filled.ArrowForwardIos +import androidx.compose.material.icons.automirrored.filled.ArrowForwardIos import androidx.compose.material.icons.filled.Check import androidx.compose.material3.Card import androidx.compose.material3.ExperimentalMaterial3Api @@ -32,25 +32,25 @@ import androidx.compose.ui.res.stringResource import androidx.compose.ui.text.font.FontWeight import androidx.compose.ui.unit.dp import androidx.compose.ui.unit.sp -import com.nexters.bandalart.android.core.ui.R -import com.nexters.bandalart.android.core.ui.component.EmojiText -import com.nexters.bandalart.android.core.ui.component.FixedSizeText -import com.nexters.bandalart.android.core.ui.extension.toColor import com.nexters.bandalart.android.core.designsystem.theme.Gray100 import com.nexters.bandalart.android.core.designsystem.theme.Gray300 import com.nexters.bandalart.android.core.designsystem.theme.Gray400 import com.nexters.bandalart.android.core.designsystem.theme.Gray900 +import com.nexters.bandalart.android.core.ui.R +import com.nexters.bandalart.android.core.ui.component.EmojiText +import com.nexters.bandalart.android.core.ui.component.FixedSizeText +import com.nexters.bandalart.android.core.ui.extension.toColor import com.nexters.bandalart.android.feature.home.model.BandalartDetailUiModel import kotlinx.coroutines.launch @Composable fun BandalartItem( - modifier: Modifier = Modifier, bottomSheetState: SheetState, bandalartItem: BandalartDetailUiModel, currentBandalartKey: String, onClick: (String) -> Unit, onCancelClicked: () -> Unit, + modifier: Modifier = Modifier, ) { val scope = rememberCoroutineScope() @@ -163,7 +163,7 @@ fun BandalartItem( if (currentBandalartKey != bandalartItem.key) { Box(modifier = Modifier.align(Alignment.CenterVertically)) { Icon( - imageVector = Icons.Default.ArrowForwardIos, + imageVector = Icons.AutoMirrored.Filled.ArrowForwardIos, contentDescription = stringResource(R.string.arrow_forward_descrption), tint = Gray400, modifier = Modifier.size(16.dp), diff --git a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartListBottomSheet.kt b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartListBottomSheet.kt index b376c141..3aa54169 100644 --- a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartListBottomSheet.kt +++ b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartListBottomSheet.kt @@ -39,8 +39,8 @@ import androidx.compose.ui.unit.dp import androidx.compose.ui.unit.sp import com.nexters.bandalart.android.core.ui.R import com.nexters.bandalart.android.core.ui.component.FixedSizeText -import com.nexters.bandalart.android.core.ui.extension.NavigationBarHeightDp -import com.nexters.bandalart.android.core.ui.extension.nonScaleSp +import com.nexters.bandalart.android.core.ui.NavigationBarHeightDp +import com.nexters.bandalart.android.core.ui.nonScaleSp import com.nexters.bandalart.android.core.designsystem.theme.Gray200 import com.nexters.bandalart.android.core.designsystem.theme.Gray600 import com.nexters.bandalart.android.core.designsystem.theme.Gray800 @@ -52,7 +52,6 @@ import kotlinx.coroutines.launch @Composable fun BandalartListBottomSheet( - modifier: Modifier = Modifier, bandalartList: ImmutableList, currentBandalartKey: String, getBandalartDetail: (String) -> Unit, @@ -60,6 +59,7 @@ fun BandalartListBottomSheet( showSkeletonChanged: (Boolean) -> Unit, onCancelClicked: () -> Unit, createBandalart: () -> Unit, + modifier: Modifier = Modifier, ) { val scope = rememberCoroutineScope() val bottomSheetState = rememberModalBottomSheetState(skipPartiallyExpanded = true) diff --git a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartSkeleton.kt b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartSkeleton.kt index f38a0b41..0a850050 100644 --- a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartSkeleton.kt +++ b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartSkeleton.kt @@ -7,6 +7,7 @@ import androidx.compose.animation.core.infiniteRepeatable import androidx.compose.animation.core.rememberInfiniteTransition import androidx.compose.animation.core.tween import androidx.compose.runtime.Composable +import androidx.compose.ui.Modifier import androidx.compose.ui.geometry.Offset import androidx.compose.ui.graphics.Brush import androidx.compose.ui.res.stringResource @@ -19,7 +20,9 @@ import com.nexters.bandalart.android.core.designsystem.theme.Gray50 import com.nexters.bandalart.android.core.designsystem.theme.White @Composable -fun BandalartSkeleton() { +fun BandalartSkeleton( + modifier: Modifier = Modifier, +) { val shimmerMainColors = listOf( Gray200, Gray300, @@ -66,5 +69,6 @@ fun BandalartSkeleton() { taskBrush = taskBrush, subBrush = subBrush, mainBrush = mainBrush, + modifier = modifier, ) } diff --git a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartSkeletonScreen.kt b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartSkeletonScreen.kt index 6d734051..d56d8e99 100644 --- a/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartSkeletonScreen.kt +++ b/feature/home/src/main/kotlin/com/nexters/bandalart/android/feature/home/ui/bandalart/BandalartSkeletonScreen.kt @@ -40,7 +40,7 @@ import androidx.compose.ui.unit.dp import androidx.compose.ui.unit.sp import com.nexters.bandalart.android.core.ui.R import com.nexters.bandalart.android.core.ui.component.FixedSizeText -import com.nexters.bandalart.android.core.ui.extension.nonScaleSp +import com.nexters.bandalart.android.core.ui.nonScaleSp import com.nexters.bandalart.android.core.designsystem.theme.Gray100 import com.nexters.bandalart.android.core.designsystem.theme.Gray200 import com.nexters.bandalart.android.core.designsystem.theme.Gray300 @@ -49,14 +49,14 @@ import com.nexters.bandalart.android.core.designsystem.theme.Gray600 import com.nexters.bandalart.android.core.designsystem.theme.Gray900 import com.nexters.bandalart.android.core.designsystem.theme.White import com.nexters.bandalart.android.core.designsystem.theme.neurimboGothicRegular -import com.nexters.bandalart.android.feature.home.ui.CompletionRatioProgressBar +import com.nexters.bandalart.android.core.ui.component.CompletionRatioProgressBar @Composable fun BandalartSkeletonScreen( - modifier: Modifier = Modifier, taskBrush: Brush, subBrush: Brush, mainBrush: Brush, + modifier: Modifier = Modifier, ) { Surface( modifier = modifier.fillMaxSize(), diff --git a/feature/onboarding/src/main/kotlin/com/nexters/bandalart/android/feature/onboarding/OnBoardingScreen.kt b/feature/onboarding/src/main/kotlin/com/nexters/bandalart/android/feature/onboarding/OnBoardingScreen.kt index 163e4dda..3c1dda69 100644 --- a/feature/onboarding/src/main/kotlin/com/nexters/bandalart/android/feature/onboarding/OnBoardingScreen.kt +++ b/feature/onboarding/src/main/kotlin/com/nexters/bandalart/android/feature/onboarding/OnBoardingScreen.kt @@ -26,6 +26,7 @@ import androidx.compose.ui.Modifier import androidx.compose.ui.platform.LocalContext import androidx.compose.ui.res.painterResource import androidx.compose.ui.unit.dp +import androidx.hilt.navigation.compose.hiltViewModel import androidx.navigation.NavOptions import com.airbnb.lottie.compose.LottieAnimation import com.airbnb.lottie.compose.LottieCompositionSpec @@ -36,19 +37,37 @@ import com.nexters.bandalart.android.core.ui.R import com.nexters.bandalart.android.core.ui.component.BandalartButton import com.nexters.bandalart.android.core.ui.component.TitleText import com.nexters.bandalart.android.core.designsystem.theme.Gray50 +import com.nexters.bandalart.android.core.ui.ObserveAsEvents import com.nexters.bandalart.android.feature.onboarding.navigation.ONBOARDING_NAVIGATION_ROUTE +import com.nexters.bandalart.android.core.ui.component.PagerIndicator @Composable internal fun OnBoardingRoute( navigateToHome: (NavOptions) -> Unit, + modifier: Modifier = Modifier, + viewModel: SplashViewModel = hiltViewModel(), ) { - OnBoardingScreen(navigateToHome = navigateToHome) + ObserveAsEvents(flow = viewModel.eventFlow) { event -> + when (event) { + is OnBoardingUiEvent.NavigateToHome -> { + val options = NavOptions.Builder() + .setPopUpTo(ONBOARDING_NAVIGATION_ROUTE, inclusive = true) + .build() + navigateToHome(options) + } + } + } + + OnBoardingScreen( + navigateToHome = viewModel::navigateToHome, + modifier = modifier, + ) } @Composable internal fun OnBoardingScreen( + navigateToHome: () -> Unit, modifier: Modifier = Modifier, - navigateToHome: (NavOptions) -> Unit, ) { val composition by rememberLottieComposition( spec = LottieCompositionSpec.RawRes( @@ -141,12 +160,7 @@ internal fun OnBoardingScreen( } } BandalartButton( - onClick = { - val options = NavOptions.Builder() - .setPopUpTo(ONBOARDING_NAVIGATION_ROUTE, inclusive = true) - .build() - navigateToHome(options) - }, + onClick = navigateToHome, text = context.getString(R.string.onboarding_start), modifier = Modifier .align(Alignment.BottomCenter) diff --git a/feature/onboarding/src/main/kotlin/com/nexters/bandalart/android/feature/onboarding/OnBoardingViewModel.kt b/feature/onboarding/src/main/kotlin/com/nexters/bandalart/android/feature/onboarding/OnBoardingViewModel.kt new file mode 100644 index 00000000..f5d47666 --- /dev/null +++ b/feature/onboarding/src/main/kotlin/com/nexters/bandalart/android/feature/onboarding/OnBoardingViewModel.kt @@ -0,0 +1,26 @@ +package com.nexters.bandalart.android.feature.onboarding + +import androidx.lifecycle.ViewModel +import androidx.lifecycle.viewModelScope +import dagger.hilt.android.lifecycle.HiltViewModel +import kotlinx.coroutines.flow.MutableSharedFlow +import kotlinx.coroutines.flow.SharedFlow +import kotlinx.coroutines.flow.asSharedFlow +import kotlinx.coroutines.launch +import javax.inject.Inject + +sealed interface OnBoardingUiEvent { + data object NavigateToHome : OnBoardingUiEvent +} + +@HiltViewModel +class SplashViewModel @Inject constructor() : ViewModel() { + private val _eventFlow = MutableSharedFlow() + val eventFlow: SharedFlow = _eventFlow.asSharedFlow() + + fun navigateToHome() { + viewModelScope.launch { + _eventFlow.emit(OnBoardingUiEvent.NavigateToHome) + } + } +} diff --git a/feature/splash/src/main/kotlin/com/nexters/bandalart/android/feature/splash/SplashScreen.kt b/feature/splash/src/main/kotlin/com/nexters/bandalart/android/feature/splash/SplashScreen.kt index c2f3f7f2..930038d8 100644 --- a/feature/splash/src/main/kotlin/com/nexters/bandalart/android/feature/splash/SplashScreen.kt +++ b/feature/splash/src/main/kotlin/com/nexters/bandalart/android/feature/splash/SplashScreen.kt @@ -15,6 +15,7 @@ import androidx.hilt.navigation.compose.hiltViewModel import androidx.lifecycle.compose.collectAsStateWithLifecycle import androidx.navigation.NavOptions import com.nexters.bandalart.android.core.designsystem.theme.Gray50 +import com.nexters.bandalart.android.core.ui.ObserveAsEvents import com.nexters.bandalart.android.core.ui.R import com.nexters.bandalart.android.core.ui.component.LoadingScreen import com.nexters.bandalart.android.core.ui.component.NetworkErrorAlertDialog @@ -24,27 +25,47 @@ import com.nexters.bandalart.android.feature.splash.navigation.SPLASH_NAVIGATION internal fun SplashRoute( navigateToOnBoarding: (NavOptions) -> Unit, navigateToHome: (NavOptions) -> Unit, + modifier: Modifier = Modifier, viewModel: SplashViewModel = hiltViewModel(), ) { val uiState by viewModel.uiState.collectAsStateWithLifecycle() + ObserveAsEvents(flow = viewModel.eventFlow) { event -> + when (event) { + is SplashUiEvent.NavigateToOnBoarding -> { + val options = NavOptions.Builder() + .setPopUpTo(SPLASH_NAVIGATION_ROUTE, inclusive = true) + .build() + navigateToOnBoarding(options) + } + + is SplashUiEvent.NavigateToHome -> { + val options = NavOptions.Builder() + .setPopUpTo(SPLASH_NAVIGATION_ROUTE, inclusive = true) + .build() + navigateToHome(options) + } + } + } + SplashScreen( uiState = uiState, - navigateToOnBoarding = navigateToOnBoarding, - navigateToHome = navigateToHome, + navigateToOnBoarding = viewModel::navigateToOnBoarding, + navigateToHome = viewModel::navigateToHome, openNetworkErrorAlertDialog = viewModel::openNetworkErrorAlertDialog, createGuestLoginToken = viewModel::createGuestLoginToken, + modifier = modifier, ) } @Composable fun SplashScreen( - modifier: Modifier = Modifier, uiState: SplashUiState, - navigateToOnBoarding: (NavOptions) -> Unit, - navigateToHome: (NavOptions) -> Unit, + navigateToOnBoarding: () -> Unit, + navigateToHome: () -> Unit, openNetworkErrorAlertDialog: (Boolean) -> Unit, createGuestLoginToken: () -> Unit, + modifier: Modifier = Modifier, ) { val context = LocalContext.current @@ -65,17 +86,11 @@ fun SplashScreen( } !uiState.isLoggedIn -> { - val options = NavOptions.Builder() - .setPopUpTo(SPLASH_NAVIGATION_ROUTE, inclusive = true) - .build() - navigateToOnBoarding(options) + navigateToOnBoarding() } uiState.isLoggedIn -> { - val options = NavOptions.Builder() - .setPopUpTo(SPLASH_NAVIGATION_ROUTE, inclusive = true) - .build() - navigateToHome(options) + navigateToHome() } } diff --git a/feature/splash/src/main/kotlin/com/nexters/bandalart/android/feature/splash/SplashViewModel.kt b/feature/splash/src/main/kotlin/com/nexters/bandalart/android/feature/splash/SplashViewModel.kt index ad57fe6a..e89286b7 100644 --- a/feature/splash/src/main/kotlin/com/nexters/bandalart/android/feature/splash/SplashViewModel.kt +++ b/feature/splash/src/main/kotlin/com/nexters/bandalart/android/feature/splash/SplashViewModel.kt @@ -7,14 +7,17 @@ import com.nexters.bandalart.android.core.domain.usecase.login.CreateGuestLoginT import com.nexters.bandalart.android.core.domain.usecase.login.GetGuestLoginTokenUseCase import com.nexters.bandalart.android.core.domain.usecase.login.SetGuestLoginTokenUseCase import dagger.hilt.android.lifecycle.HiltViewModel -import javax.inject.Inject import kotlinx.coroutines.delay +import kotlinx.coroutines.flow.MutableSharedFlow import kotlinx.coroutines.flow.MutableStateFlow +import kotlinx.coroutines.flow.SharedFlow import kotlinx.coroutines.flow.StateFlow +import kotlinx.coroutines.flow.asSharedFlow import kotlinx.coroutines.flow.asStateFlow import kotlinx.coroutines.flow.update import kotlinx.coroutines.launch import timber.log.Timber +import javax.inject.Inject /** * SplashUiState @@ -32,6 +35,11 @@ data class SplashUiState( val error: Throwable? = null, ) +sealed interface SplashUiEvent { + data object NavigateToOnBoarding : SplashUiEvent + data object NavigateToHome : SplashUiEvent +} + @HiltViewModel class SplashViewModel @Inject constructor( private val getGuestLoginTokenUseCase: GetGuestLoginTokenUseCase, @@ -43,6 +51,9 @@ class SplashViewModel @Inject constructor( private val _uiState = MutableStateFlow(SplashUiState()) val uiState: StateFlow = _uiState.asStateFlow() + private val _eventFlow = MutableSharedFlow() + val eventFlow: SharedFlow = _eventFlow.asSharedFlow() + init { viewModelScope.launch { delay(500) @@ -80,9 +91,11 @@ class SplashViewModel @Inject constructor( _uiState.update { it.copy(isLoggedIn = false) } createBandalartUseCase() } + result.isSuccess && result.getOrNull() == null -> { Timber.e("Request succeeded but data validation failed") } + result.isFailure -> { _uiState.update { it.copy(isNetworkErrorAlertDialogOpened = true) } } @@ -94,4 +107,16 @@ class SplashViewModel @Inject constructor( fun openNetworkErrorAlertDialog(flag: Boolean) { _uiState.update { it.copy(isNetworkErrorAlertDialogOpened = flag) } } + + fun navigateToOnBoarding() { + viewModelScope.launch { + _eventFlow.emit(SplashUiEvent.NavigateToOnBoarding) + } + } + + fun navigateToHome() { + viewModelScope.launch { + _eventFlow.emit(SplashUiEvent.NavigateToHome) + } + } } diff --git a/settings.gradle.kts b/settings.gradle.kts index b7802173..5f8ba983 100644 --- a/settings.gradle.kts +++ b/settings.gradle.kts @@ -28,6 +28,7 @@ include( ":core:domain", ":core:network", ":core:ui", + ":core:util", ":feature:complete", ":feature:home", ":feature:onboarding",