diff --git a/.github/workflows/android_ci.yml b/.github/workflows/android_ci.yml index 1f1a7173..3e073b75 100644 --- a/.github/workflows/android_ci.yml +++ b/.github/workflows/android_ci.yml @@ -7,13 +7,6 @@ on: - '**/*.kt' - 'build.gradle' - 'app/**' - push: - branches: - - 'release/**' - paths: - - '**/*.kt' - - 'build.gradle' - - 'app/**' jobs: build: diff --git a/app/build.gradle.kts b/app/build.gradle.kts index b955d234..6c350ede 100644 --- a/app/build.gradle.kts +++ b/app/build.gradle.kts @@ -8,10 +8,11 @@ plugins { android { namespace = "com.yapp.orbit" + compileSdk = 35 defaultConfig { - versionCode = 5 - versionName = "1.0.3" + versionCode = 6 + versionName = "1.1.3" targetSdk = 35 } @@ -53,4 +54,6 @@ dependencies { implementation(libs.firebase.crashlytics) implementation(libs.play.services.ads) implementation(libs.kotlin.reflect) + implementation(libs.hilt.worker) + implementation(libs.androidx.work.runtime) } diff --git a/app/src/main/AndroidManifest.xml b/app/src/main/AndroidManifest.xml index 780b61bb..e7abd75a 100644 --- a/app/src/main/AndroidManifest.xml +++ b/app/src/main/AndroidManifest.xml @@ -11,6 +11,7 @@ + - + @@ -80,5 +79,15 @@ + + + + diff --git a/app/src/main/java/com/yapp/orbit/OrbitApplication.kt b/app/src/main/java/com/yapp/orbit/OrbitApplication.kt index 7391cf6e..b06538f8 100644 --- a/app/src/main/java/com/yapp/orbit/OrbitApplication.kt +++ b/app/src/main/java/com/yapp/orbit/OrbitApplication.kt @@ -1,13 +1,24 @@ package com.yapp.orbit import android.app.Application +import androidx.hilt.work.HiltWorkerFactory +import androidx.work.Configuration import com.google.android.gms.ads.MobileAds import dagger.hilt.android.HiltAndroidApp +import javax.inject.Inject @HiltAndroidApp -class OrbitApplication : Application() { +class OrbitApplication() : Application(), Configuration.Provider { + + @Inject lateinit var workerFactory: HiltWorkerFactory + override fun onCreate() { super.onCreate() MobileAds.initialize(this) } + + override val workManagerConfiguration: Configuration + get() = Configuration.Builder() + .setWorkerFactory(workerFactory) + .build() } diff --git a/app/src/main/java/com/yapp/orbit/OrbitNavHost.kt b/app/src/main/java/com/yapp/orbit/OrbitNavHost.kt index 69bc23ac..5dd9ce4b 100644 --- a/app/src/main/java/com/yapp/orbit/OrbitNavHost.kt +++ b/app/src/main/java/com/yapp/orbit/OrbitNavHost.kt @@ -6,24 +6,16 @@ import androidx.compose.animation.fadeIn import androidx.compose.animation.fadeOut import androidx.compose.animation.slideInVertically import androidx.compose.animation.slideOutVertically -import androidx.compose.foundation.background import androidx.compose.foundation.layout.Box -import androidx.compose.foundation.layout.BoxScope -import androidx.compose.foundation.layout.WindowInsets -import androidx.compose.foundation.layout.fillMaxWidth -import androidx.compose.foundation.layout.navigationBars +import androidx.compose.foundation.layout.navigationBarsPadding import androidx.compose.foundation.layout.padding -import androidx.compose.foundation.layout.windowInsetsBottomHeight import androidx.compose.material3.Scaffold import androidx.compose.material3.SnackbarHost import androidx.compose.material3.SnackbarHostState import androidx.compose.runtime.Composable import androidx.compose.runtime.remember -import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier -import androidx.compose.ui.graphics.Color import androidx.compose.ui.unit.dp -import androidx.compose.ui.zIndex import androidx.navigation.compose.NavHost import com.yapp.common.navigation.OrbitNavigator import com.yapp.common.navigation.rememberOrbitNavigator @@ -34,9 +26,9 @@ import com.yapp.mission.missionScreen import com.yapp.onboarding.onboardingNavGraph import com.yapp.setting.settingNavGraph import com.yapp.splash.splashScreen -import com.yapp.ui.component.bottomsheet.OrbitBottomSheetLayout import com.yapp.ui.component.bottomsheet.OrbitBottomSheetState import com.yapp.ui.component.bottomsheet.rememberOrbitBottomSheetState +import com.yapp.ui.component.navigation.NavigationBarScrim import com.yapp.ui.component.snackbar.CustomSnackBarVisuals import com.yapp.ui.component.snackbar.OrbitSnackBar import com.yapp.webview.webViewScreen @@ -51,18 +43,16 @@ internal fun OrbitNavHost( val snackBarHostState = remember { SnackbarHostState() } Box { - OrbitBottomSheetLayout(sheetState = bottomSheetState) { - Scaffold( - modifier = modifier, - snackbarHost = { OrbitSnackBarHost(snackBarHostState) }, - containerColor = OrbitTheme.colors.gray_900, - ) { - OrbitNavigationGraph( - navigator = navigator, - bottomSheetState = bottomSheetState, - snackBarHostState = snackBarHostState, - ) - } + Scaffold( + modifier = modifier, + snackbarHost = { OrbitSnackBarHost(snackBarHostState) }, + containerColor = OrbitTheme.colors.gray_900, + ) { + OrbitNavigationGraph( + navigator = navigator, + bottomSheetState = bottomSheetState, + snackBarHostState = snackBarHostState, + ) } NavigationBarScrim() @@ -76,6 +66,7 @@ private fun OrbitNavigationGraph( snackBarHostState: SnackbarHostState, ) { NavHost( + modifier = Modifier.navigationBarsPadding(), navController = navigator.navController, startDestination = navigator.startDestination, ) { @@ -89,18 +80,6 @@ private fun OrbitNavigationGraph( } } -@Composable -private fun BoxScope.NavigationBarScrim() { - Box( - modifier = Modifier - .align(Alignment.BottomCenter) - .fillMaxWidth() - .windowInsetsBottomHeight(WindowInsets.navigationBars) - .background(Color.Black) - .zIndex(1f), - ) -} - @Composable private fun OrbitSnackBarHost( snackBarHostState: SnackbarHostState, diff --git a/app/src/main/java/com/yapp/orbit/di/AppVersionModule.kt b/app/src/main/java/com/yapp/orbit/di/AppVersionModule.kt new file mode 100644 index 00000000..6297ecc3 --- /dev/null +++ b/app/src/main/java/com/yapp/orbit/di/AppVersionModule.kt @@ -0,0 +1,18 @@ +package com.yapp.orbit.di + +import com.yapp.orbit.BuildConfig +import dagger.Module +import dagger.Provides +import dagger.hilt.InstallIn +import dagger.hilt.components.SingletonComponent +import javax.inject.Named +import javax.inject.Singleton + +@Module +@InstallIn(SingletonComponent::class) +object AppVersionModule { + @Provides + @Singleton + @Named("appVersion") + fun provideAppVersion(): String = BuildConfig.VERSION_NAME +} diff --git a/build-logic/src/main/java/com/yapp/convention/HiltAndroid.kt b/build-logic/src/main/java/com/yapp/convention/HiltAndroid.kt index e266704f..15db09ff 100644 --- a/build-logic/src/main/java/com/yapp/convention/HiltAndroid.kt +++ b/build-logic/src/main/java/com/yapp/convention/HiltAndroid.kt @@ -14,7 +14,9 @@ internal fun Project.configureHiltAndroid() { dependencies { "implementation"(libs.findLibrary("hilt.android").get()) "ksp"(libs.findLibrary("hilt.android.compiler").get()) + "ksp"(libs.findLibrary("androidx-hilt-compiler").get()) "implementation"(libs.findLibrary("hilt-navigation-compose").get()) + "implementation"(libs.findLibrary("hilt-worker").get()) } } diff --git a/build-logic/src/main/java/orbit.android.feature.gradle.kts b/build-logic/src/main/java/orbit.android.feature.gradle.kts index aa5803c4..47d5c072 100644 --- a/build-logic/src/main/java/orbit.android.feature.gradle.kts +++ b/build-logic/src/main/java/orbit.android.feature.gradle.kts @@ -1,4 +1,3 @@ -import com.yapp.convention.configureHiltAndroid import com.yapp.convention.libs plugins { @@ -6,8 +5,6 @@ plugins { id("orbit.android.compose") } -configureHiltAndroid() - dependencies { implementation(project(":core:designsystem")) implementation(project(":core:ui")) diff --git a/core/alarm/src/main/java/com/yapp/alarm/AndroidAlarmScheduler.kt b/core/alarm/src/main/java/com/yapp/alarm/AndroidAlarmScheduler.kt index 2691e78b..49c4581a 100644 --- a/core/alarm/src/main/java/com/yapp/alarm/AndroidAlarmScheduler.kt +++ b/core/alarm/src/main/java/com/yapp/alarm/AndroidAlarmScheduler.kt @@ -2,6 +2,7 @@ package com.yapp.alarm import android.app.AlarmManager import android.app.Application +import android.util.Log import com.yapp.alarm.pendingIntent.schedule.createAlarmReceiverPendingIntentForSchedule import com.yapp.alarm.pendingIntent.schedule.createAlarmReceiverPendingIntentForUnSchedule import com.yapp.domain.model.Alarm @@ -16,6 +17,15 @@ class AndroidAlarmScheduler @Inject constructor( private val alarmTimeCalculator: AlarmTimeCalculator, ) : AlarmScheduler { + private fun logSchedule(tag: String, alarm: Alarm, triggerMillis: Long, extra: String = "") { + Log.d("ScheduleTrace", "scheduleAlarm Called", Throwable()) + Log.d( + "AlarmSchedule", + "[$tag] id=${alarm.id}, repeatDays=${alarm.repeatDays}, " + + "time=${java.time.Instant.ofEpochMilli(triggerMillis)} $extra", + ) + } + override fun scheduleAlarm(alarm: Alarm) { val selectedDays = alarm.repeatDays.toAlarmDays() @@ -31,7 +41,7 @@ class AndroidAlarmScheduler @Inject constructor( private fun setRepeatingAlarm(day: AlarmDay, alarm: Alarm) { val triggerMillis = alarmTimeCalculator.calculateNextRepeatingTimeMillis(alarm, day) val pendingIntent = createAlarmReceiverPendingIntentForSchedule(app, alarm, day) - + logSchedule("REPEAT", alarm, triggerMillis, "day=$day") alarmManager.setExactAndAllowWhileIdle( AlarmManager.RTC_WAKEUP, triggerMillis, @@ -42,7 +52,7 @@ class AndroidAlarmScheduler @Inject constructor( private fun setNonRepeatingAlarm(alarm: Alarm) { val triggerMillis = alarmTimeCalculator.calculateNonRepeatingTimeMillis(alarm) val pendingIntent = createAlarmReceiverPendingIntentForSchedule(app, alarm) - + logSchedule("NON_REPEAT", alarm, triggerMillis) alarmManager.setExactAndAllowWhileIdle( AlarmManager.RTC_WAKEUP, triggerMillis, @@ -53,7 +63,7 @@ class AndroidAlarmScheduler @Inject constructor( fun rescheduleUpcomingWeeklyAlarm(alarm: Alarm, day: AlarmDay) { val triggerMillis = alarmTimeCalculator.calculateNextWeeklyRescheduledTimeMillis(alarm, day) val pendingIntent = createAlarmReceiverPendingIntentForSchedule(app, alarm, day) - + logSchedule("RESCHEDULE_WEEKLY", alarm, triggerMillis, "day=$day") alarmManager.setExactAndAllowWhileIdle( AlarmManager.RTC_WAKEUP, triggerMillis, diff --git a/core/alarm/src/main/java/com/yapp/alarm/receivers/AlarmInteractionActivityReceiver.kt b/core/alarm/src/main/java/com/yapp/alarm/receivers/AlarmInteractionActivityReceiver.kt index d7f38c3e..029072a4 100644 --- a/core/alarm/src/main/java/com/yapp/alarm/receivers/AlarmInteractionActivityReceiver.kt +++ b/core/alarm/src/main/java/com/yapp/alarm/receivers/AlarmInteractionActivityReceiver.kt @@ -3,18 +3,18 @@ package com.yapp.alarm.receivers import android.content.BroadcastReceiver import android.content.Context import android.content.Intent -import android.util.Log import androidx.activity.ComponentActivity import androidx.core.net.toUri import com.yapp.alarm.AlarmConstants +import com.yapp.domain.model.FortuneCreateStatus +import com.yapp.domain.model.MissionType import com.yapp.domain.repository.FortuneRepository import dagger.hilt.android.AndroidEntryPoint import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.Dispatchers -import kotlinx.coroutines.flow.firstOrNull +import kotlinx.coroutines.flow.first import kotlinx.coroutines.launch -import java.time.LocalDate -import java.time.format.DateTimeFormatter +import kotlinx.coroutines.withContext import javax.inject.Inject @AndroidEntryPoint @@ -31,29 +31,69 @@ class AlarmInteractionActivityReceiver(private val activity: ComponentActivity) if (!isSnoozed) { val notificationId = intent.getLongExtra(AlarmConstants.EXTRA_NOTIFICATION_ID, -1L) - val missionType = intent.getIntExtra(AlarmConstants.EXTRA_MISSION_TYPE, -1) + val missionTypeRaw = intent.getIntExtra(AlarmConstants.EXTRA_MISSION_TYPE, -1) val missionCount = intent.getIntExtra(AlarmConstants.EXTRA_MISSION_COUNT, -1) - if (notificationId == -1L || missionType == -1 || missionCount == -1) { - Log.e("AlarmInteraction", "필수 값 누락") - return - } + val missionType = MissionType.fromInt(missionTypeRaw) + + val hasValidMissionData = ( + notificationId != -1L && + missionType != MissionType.NONE && + missionCount != -1 + ) + + val pending = goAsync() + CoroutineScope(Dispatchers.Main).launch { + try { + if (!hasValidMissionData) { + val (fortuneCreateStatus, hasUnseenFortune) = withContext(Dispatchers.IO) { + val status = fortuneRepository.fortuneCreateStatusFlow.first() + val unseen = fortuneRepository.hasUnseenFortuneFlow.first() + status to unseen + } - CoroutineScope(Dispatchers.IO).launch { - val fortuneDate = fortuneRepository.fortuneDateFlow.firstOrNull() - val todayDate = LocalDate.now().format(DateTimeFormatter.ISO_DATE) - - if (fortuneDate != todayDate) { - context?.let { - val uriString = - "orbitapp://mission?notificationId=$notificationId&missionType=$missionType&missionCount=$missionCount" - val missionIntent = - Intent(Intent.ACTION_VIEW, uriString.toUri()).apply { - addFlags(Intent.FLAG_ACTIVITY_NEW_TASK) - setPackage(context.packageName) + when (fortuneCreateStatus) { + is FortuneCreateStatus.Creating -> { + context?.let { ctx -> + val uri = "orbitapp://fortune".toUri() + val fortuneIntent = Intent(Intent.ACTION_VIEW, uri).apply { + addFlags(Intent.FLAG_ACTIVITY_NEW_TASK) + setPackage(ctx.packageName) + } + ctx.startActivity(fortuneIntent) + } } - it.startActivity(missionIntent) + + is FortuneCreateStatus.Success -> { + if (hasUnseenFortune) { + context?.let { ctx -> + val uri = "orbitapp://fortune".toUri() + val fortuneIntent = + Intent(Intent.ACTION_VIEW, uri).apply { + addFlags(Intent.FLAG_ACTIVITY_NEW_TASK) + setPackage(ctx.packageName) + } + ctx.startActivity(fortuneIntent) + } + } + } + + FortuneCreateStatus.Failure, FortuneCreateStatus.Idle -> { } + } + } else { + context?.let { ctx -> + val uriString = + "orbitapp://mission?notificationId=$notificationId&missionType=${missionType.value}&missionCount=$missionCount" + val missionIntent = + Intent(Intent.ACTION_VIEW, uriString.toUri()).apply { + addFlags(Intent.FLAG_ACTIVITY_NEW_TASK) + setPackage(ctx.packageName) + } + ctx.startActivity(missionIntent) + } } + } finally { + pending.finish() } } } diff --git a/core/alarm/src/main/java/com/yapp/alarm/receivers/AlarmReceiver.kt b/core/alarm/src/main/java/com/yapp/alarm/receivers/AlarmReceiver.kt index edd18466..011dadef 100644 --- a/core/alarm/src/main/java/com/yapp/alarm/receivers/AlarmReceiver.kt +++ b/core/alarm/src/main/java/com/yapp/alarm/receivers/AlarmReceiver.kt @@ -12,6 +12,7 @@ import com.yapp.alarm.services.AlarmService import com.yapp.analytics.AnalyticsEvent import com.yapp.analytics.AnalyticsHelper import com.yapp.domain.model.Alarm +import com.yapp.domain.model.toAlarmDay import com.yapp.domain.model.toTimeString import com.yapp.domain.repository.FortuneRepository import com.yapp.domain.usecase.AlarmUseCase @@ -19,8 +20,8 @@ import dagger.hilt.android.AndroidEntryPoint import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.flow.first -import kotlinx.coroutines.flow.firstOrNull import kotlinx.coroutines.launch +import java.time.LocalDate import java.time.LocalDateTime import javax.inject.Inject @@ -105,17 +106,31 @@ class AlarmReceiver : BroadcastReceiver() { androidAlarmScheduler.cancelSnoozedAlarm(notificationId) context.stopService(alarmServiceIntent) - sendBroadCastToCloseAlarmInteractionActivity( - context = context, - notificationId = notificationId, - missionType = missionType, - missionCount = missionCount, - ) - CoroutineScope(Dispatchers.IO).launch { - val alarms = alarmUseCase.getAllAlarms().first().sortedBy { it.isAlarmActive } - val isFirstAlarm = alarms.firstOrNull()?.id == notificationId + val alarms = alarmUseCase.getAllAlarms().first() + + val isSnoozeId = notificationId >= AlarmConstants.SNOOZE_ID_OFFSET + + fun Alarm.ringsToday(): Boolean { + if (repeatDays == 0) return true + + val todayAlarmDay = LocalDate.now().dayOfWeek.toAlarmDay() + return (repeatDays and todayAlarmDay.bitValue) != 0 + } + val earliestIdToday: Long? = alarms + .asSequence() + .filter { (it.isAlarmActive || it.id == notificationId) && it.ringsToday() } + .sortedWith(compareBy({ it.hour }, { it.minute }, { it.second })) + .firstOrNull() + ?.id + + val isEarliestAlarmDismissedToday = + !isSnoozeId && (earliestIdToday == notificationId) + + if (isEarliestAlarmDismissedToday) fortuneRepository.markFirstAlarmDismissedToday() + + val isFirstAlarm = earliestIdToday == notificationId analyticsHelper.logEvent( AnalyticsEvent( type = "alarm_dismiss", @@ -126,12 +141,12 @@ class AlarmReceiver : BroadcastReceiver() { ), ) - val existingId = fortuneRepository.firstDismissedAlarmIdFlow.firstOrNull() - if (existingId == null) { - fortuneRepository.saveFirstDismissedAlarmId(notificationId) - } else if (existingId != notificationId) { - fortuneRepository.clearDismissedAlarmId() - } + sendBroadCastToCloseAlarmInteractionActivity( + context = context, + notificationId = notificationId, + missionType = missionType, + missionCount = missionCount, + ) } Toast.makeText(context, "알람이 해제되었어요", Toast.LENGTH_SHORT).show() diff --git a/core/alarm/src/main/java/com/yapp/alarm/receivers/RescheduleAlarmReceiver.kt b/core/alarm/src/main/java/com/yapp/alarm/receivers/RescheduleAlarmReceiver.kt index 9d21144d..bdd47fe8 100644 --- a/core/alarm/src/main/java/com/yapp/alarm/receivers/RescheduleAlarmReceiver.kt +++ b/core/alarm/src/main/java/com/yapp/alarm/receivers/RescheduleAlarmReceiver.kt @@ -8,6 +8,8 @@ import com.yapp.domain.usecase.AlarmUseCase import dagger.hilt.android.AndroidEntryPoint import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.SupervisorJob +import kotlinx.coroutines.flow.first import kotlinx.coroutines.launch import javax.inject.Inject @@ -25,16 +27,20 @@ class RescheduleAlarmReceiver : BroadcastReceiver() { intent ?: return if (intent.action == Intent.ACTION_BOOT_COMPLETED) { - rescheduleAlarm() + val pending = goAsync() + rescheduleAlarm(pending) } } - private fun rescheduleAlarm() { - CoroutineScope(Dispatchers.IO).launch { - alarmUseCase.getAllAlarms().collect { alarms -> - alarms.forEach { alarm -> - androidAlarmScheduler.scheduleAlarm(alarm) - } + private fun rescheduleAlarm(pendingResult: PendingResult) { + CoroutineScope(Dispatchers.IO + SupervisorJob()).launch { + try { + val alarms = alarmUseCase.getAllAlarms().first() + alarms + .filter { it.isAlarmActive } + .forEach { alarm -> androidAlarmScheduler.scheduleAlarm(alarm) } + } finally { + pendingResult.finish() } } } diff --git a/core/alarm/src/main/java/com/yapp/alarm/scheduler/PostFortuneTaskScheduler.kt b/core/alarm/src/main/java/com/yapp/alarm/scheduler/PostFortuneTaskScheduler.kt new file mode 100644 index 00000000..e16e8dd4 --- /dev/null +++ b/core/alarm/src/main/java/com/yapp/alarm/scheduler/PostFortuneTaskScheduler.kt @@ -0,0 +1,5 @@ +package com.yapp.alarm.scheduler + +interface PostFortuneTaskScheduler { + fun enqueueOnceForToday() +} diff --git a/core/alarm/src/main/java/com/yapp/alarm/services/AlarmService.kt b/core/alarm/src/main/java/com/yapp/alarm/services/AlarmService.kt index cbea504e..56141679 100644 --- a/core/alarm/src/main/java/com/yapp/alarm/services/AlarmService.kt +++ b/core/alarm/src/main/java/com/yapp/alarm/services/AlarmService.kt @@ -23,9 +23,10 @@ import com.yapp.alarm.pendingIntent.interaction.createAlarmAlertPendingIntent import com.yapp.alarm.pendingIntent.interaction.createAlarmDismissPendingIntent import com.yapp.alarm.pendingIntent.interaction.createAlarmSnoozePendingIntent import com.yapp.alarm.pendingIntent.interaction.createNavigateToMissionPendingIntent +import com.yapp.alarm.scheduler.PostFortuneTaskScheduler import com.yapp.domain.model.Alarm import com.yapp.domain.model.AlarmDay -import com.yapp.domain.repository.FortuneRepository +import com.yapp.domain.model.MissionType import com.yapp.domain.usecase.AlarmUseCase import com.yapp.media.sound.SoundPlayer import dagger.hilt.android.AndroidEntryPoint @@ -33,10 +34,7 @@ import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.SupervisorJob import kotlinx.coroutines.cancel -import kotlinx.coroutines.flow.firstOrNull import kotlinx.coroutines.launch -import java.time.LocalDate -import java.time.format.DateTimeFormatter import javax.inject.Inject @AndroidEntryPoint @@ -54,7 +52,7 @@ class AlarmService : Service() { lateinit var androidAlarmScheduler: AndroidAlarmScheduler @Inject - lateinit var fortuneRepository: FortuneRepository + lateinit var postFortuneTaskScheduler: PostFortuneTaskScheduler private val serviceScope = CoroutineScope(Dispatchers.IO + SupervisorJob()) @@ -81,7 +79,7 @@ class AlarmService : Service() { super.onDestroy() } - private suspend fun handleIntent(intent: Intent) { + private fun handleIntent(intent: Intent) { val alarm: Alarm? = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) { intent.getParcelableExtra(AlarmConstants.EXTRA_ALARM, Alarm::class.java) } else { @@ -113,7 +111,7 @@ class AlarmService : Service() { false -> { startForeground( notificationId.toInt(), - createNotification(alarm, shouldNavigateToMission()), + createNotification(alarm, shouldNavigateToMission(alarm.missionType)), ) if (alarm.isVibrationEnabled) startVibration() if (alarm.isSoundEnabled) startSound(alarm.soundUri, alarm.soundVolume) @@ -123,12 +121,14 @@ class AlarmService : Service() { if (isOneTimeAlarm) { turnOffAlarm(alarmId = notificationId) } + + postFortuneTaskScheduler.enqueueOnceForToday() } - private suspend fun shouldNavigateToMission(): Boolean { - val fortuneDate = fortuneRepository.fortuneDateFlow.firstOrNull() - val todayDate = LocalDate.now().format(DateTimeFormatter.ISO_DATE) - return fortuneDate != todayDate + private fun shouldNavigateToMission( + missionType: MissionType, + ): Boolean { + return missionType != MissionType.NONE } private fun createNotification(alarm: Alarm, shouldNavigateToMission: Boolean): Notification { diff --git a/core/common/src/main/java/com/yapp/common/navigation/route/MissionRoute.kt b/core/common/src/main/java/com/yapp/common/navigation/route/MissionRoute.kt index 2ffd29d6..111c7788 100644 --- a/core/common/src/main/java/com/yapp/common/navigation/route/MissionRoute.kt +++ b/core/common/src/main/java/com/yapp/common/navigation/route/MissionRoute.kt @@ -8,8 +8,4 @@ data class MissionRoute( val missionType: String, val missionCount: String, val missionMode: String = MissionMode.REAL.name, -) { - companion object { - const val route = "mission" - } -} +) diff --git a/core/datastore/src/main/java/com/yapp/datastore/UserPreferences.kt b/core/datastore/src/main/java/com/yapp/datastore/UserPreferences.kt index 35298556..c9325870 100644 --- a/core/datastore/src/main/java/com/yapp/datastore/UserPreferences.kt +++ b/core/datastore/src/main/java/com/yapp/datastore/UserPreferences.kt @@ -1,6 +1,5 @@ package com.yapp.datastore -import android.util.Log import androidx.datastore.core.DataStore import androidx.datastore.preferences.core.Preferences import androidx.datastore.preferences.core.booleanPreferencesKey @@ -14,7 +13,6 @@ import kotlinx.coroutines.flow.catch import kotlinx.coroutines.flow.distinctUntilChanged import kotlinx.coroutines.flow.map import java.time.LocalDate -import java.time.format.DateTimeFormatter import javax.inject.Inject import javax.inject.Singleton @@ -26,15 +24,25 @@ class UserPreferences @Inject constructor( val USER_ID = longPreferencesKey("user_id") val USER_NAME = stringPreferencesKey("user_name") val ONBOARDING_COMPLETED = booleanPreferencesKey("onboarding_completed") + val FORTUNE_ID = longPreferencesKey("fortune_id") - val FORTUNE_DATE = stringPreferencesKey("fortune_date") + val FORTUNE_DATE_EPOCH = longPreferencesKey("fortune_date_epoch") val FORTUNE_IMAGE_ID = intPreferencesKey("fortune_image_id") val FORTUNE_SCORE = intPreferencesKey("fortune_score") - val FORTUNE_CHECKED = booleanPreferencesKey("fortune_checked") - val FIRST_DISMISSED_ALARM_ID = longPreferencesKey("first_dismissed_alarm_id") - val DISMISSED_DATE = stringPreferencesKey("dismissed_date") + val FORTUNE_SEEN = booleanPreferencesKey("fortune_seen") + val FORTUNE_TOOLTIP_SHOWN = booleanPreferencesKey("fortune_tooltip_shown") + val FORTUNE_CREATING = booleanPreferencesKey("fortune_creating") + val FORTUNE_FAILED = booleanPreferencesKey("fortune_failed") + + val FIRST_ALARM_DISMISSED_TODAY = booleanPreferencesKey("first_alarm_dismissed_today") + val FIRST_ALARM_DISMISSED_DATE_EPOCH = longPreferencesKey("first_alarm_dismissed_date_epoch") + + val UPDATE_NOTICE_DONT_SHOW_VERSION = stringPreferencesKey("update_notice_dont_show_version") + val UPDATE_NOTICE_LAST_SHOWN_DATE_EPOCH = longPreferencesKey("update_notice_last_shown_date_epoch") } + private fun todayEpoch(): Long = LocalDate.now().toEpochDay() + val userIdFlow: Flow = dataStore.data .catch { emit(emptyPreferences()) } .map { it[Keys.USER_ID] } @@ -55,9 +63,9 @@ class UserPreferences @Inject constructor( .map { it[Keys.FORTUNE_ID] } .distinctUntilChanged() - val fortuneDateFlow: Flow = dataStore.data + val fortuneDateEpochFlow: Flow = dataStore.data .catch { emit(emptyPreferences()) } - .map { it[Keys.FORTUNE_DATE] } + .map { it[Keys.FORTUNE_DATE_EPOCH] } .distinctUntilChanged() val fortuneImageIdFlow: Flow = dataStore.data @@ -70,108 +78,143 @@ class UserPreferences @Inject constructor( .map { it[Keys.FORTUNE_SCORE] } .distinctUntilChanged() - val hasNewFortuneFlow: Flow = dataStore.data + val hasUnseenFortuneFlow: Flow = dataStore.data .catch { emit(emptyPreferences()) } - .map { preferences -> - val savedDate = preferences[Keys.FORTUNE_DATE] - val isChecked = preferences[Keys.FORTUNE_CHECKED] ?: true - val todayDate = LocalDate.now().format(DateTimeFormatter.ISO_DATE) - savedDate == todayDate && !isChecked + .map { pref -> + val isToday = pref[Keys.FORTUNE_DATE_EPOCH] == todayEpoch() + isToday && (pref[Keys.FORTUNE_ID] != null) && (pref[Keys.FORTUNE_SEEN] != true) } .distinctUntilChanged() - val firstDismissedAlarmIdFlow: Flow = dataStore.data + val shouldShowFortuneToolTipFlow: Flow = dataStore.data .catch { emit(emptyPreferences()) } - .map { preferences -> - val savedDate = preferences[Keys.DISMISSED_DATE] - val todayDate = LocalDate.now().format(DateTimeFormatter.ISO_DATE) - - if (savedDate == todayDate) { - preferences[Keys.FIRST_DISMISSED_ALARM_ID] - } else { - null - } + .map { pref -> + val hasTodayFortune = (pref[Keys.FORTUNE_DATE_EPOCH] == todayEpoch()) && (pref[Keys.FORTUNE_ID] != null) + val tooltipShown = pref[Keys.FORTUNE_TOOLTIP_SHOWN] ?: false + hasTodayFortune && !tooltipShown } .distinctUntilChanged() - suspend fun saveUserId(userId: Long) { - dataStore.edit { preferences -> - preferences[Keys.USER_ID] = userId + val isFortuneCreatingFlow: Flow = dataStore.data + .catch { emit(emptyPreferences()) } + .map { it[Keys.FORTUNE_CREATING] ?: false } + .distinctUntilChanged() + + val isFortuneFailedFlow: Flow = dataStore.data + .catch { emit(emptyPreferences()) } + .map { it[Keys.FORTUNE_FAILED] ?: false } + .distinctUntilChanged() + + val isFirstAlarmDismissedTodayFlow: Flow = dataStore.data + .catch { emit(emptyPreferences()) } + .map { pref -> + val flag = pref[Keys.FIRST_ALARM_DISMISSED_TODAY] ?: false + val isToday = pref[Keys.FIRST_ALARM_DISMISSED_DATE_EPOCH] == todayEpoch() + flag && isToday } + .distinctUntilChanged() + + val updateNoticeDontShowVersionFlow: Flow = dataStore.data + .catch { emit(emptyPreferences()) } + .map { it[Keys.UPDATE_NOTICE_DONT_SHOW_VERSION] } + .distinctUntilChanged() + + val updateNoticeLastShownDateEpochFlow: Flow = dataStore.data + .catch { emit(emptyPreferences()) } + .map { it[Keys.UPDATE_NOTICE_LAST_SHOWN_DATE_EPOCH] } + .distinctUntilChanged() + + suspend fun saveUserId(userId: Long) { + dataStore.edit { it[Keys.USER_ID] = userId } } suspend fun saveUserName(userName: String) { - dataStore.edit { preferences -> - preferences[Keys.USER_NAME] = userName + dataStore.edit { it[Keys.USER_NAME] = userName } + } + + suspend fun setOnboardingCompleted() { + dataStore.edit { it[Keys.ONBOARDING_COMPLETED] = true } + } + + suspend fun markFortuneCreating() { + dataStore.edit { pref -> + pref[Keys.FORTUNE_CREATING] = true + pref[Keys.FORTUNE_FAILED] = false } } - suspend fun saveFortuneId(fortuneId: Long) { - val currentDate = LocalDate.now().format(DateTimeFormatter.ISO_DATE) - dataStore.edit { preferences -> - preferences[Keys.FORTUNE_ID] = fortuneId - preferences[Keys.FORTUNE_DATE] = currentDate - preferences[Keys.FORTUNE_CHECKED] = false + suspend fun markFortuneCreated(fortuneId: Long) { + dataStore.edit { pref -> + val today = todayEpoch() + val prevDate = pref[Keys.FORTUNE_DATE_EPOCH] + val isNewForToday = (pref[Keys.FORTUNE_ID] != fortuneId) || (prevDate != today) + + pref[Keys.FORTUNE_ID] = fortuneId + pref[Keys.FORTUNE_DATE_EPOCH] = today + pref[Keys.FORTUNE_CREATING] = false + pref[Keys.FORTUNE_FAILED] = false + + if (isNewForToday) { + pref[Keys.FORTUNE_SEEN] = false + pref[Keys.FORTUNE_TOOLTIP_SHOWN] = false + } } } - suspend fun markFortuneAsChecked() { - dataStore.edit { preferences -> - preferences[Keys.FORTUNE_CHECKED] = true + suspend fun markFortuneFailed() { + dataStore.edit { pref -> + pref[Keys.FORTUNE_CREATING] = false + pref[Keys.FORTUNE_FAILED] = true } } + suspend fun markFortuneSeen() { + dataStore.edit { it[Keys.FORTUNE_SEEN] = true } + } + + suspend fun markFortuneTooltipShown() { + dataStore.edit { it[Keys.FORTUNE_TOOLTIP_SHOWN] = true } + } + suspend fun saveFortuneImageId(imageResId: Int) { - dataStore.edit { preferences -> - preferences[Keys.FORTUNE_IMAGE_ID] = imageResId - } + dataStore.edit { it[Keys.FORTUNE_IMAGE_ID] = imageResId } } suspend fun saveFortuneScore(score: Int) { - dataStore.edit { preferences -> - preferences[Keys.FORTUNE_SCORE] = score - } + dataStore.edit { it[Keys.FORTUNE_SCORE] = score } } - suspend fun saveFirstDismissedAlarmId(alarmId: Long) { - dataStore.edit { preferences -> - val todayDate = LocalDate.now().format(DateTimeFormatter.ISO_DATE) - if (preferences[Keys.FIRST_DISMISSED_ALARM_ID] == null) { - preferences[Keys.FIRST_DISMISSED_ALARM_ID] = alarmId - preferences[Keys.DISMISSED_DATE] = todayDate - Log.d("UserPreferences", "첫 해제된 알람 ID 저장 완료: $alarmId (날짜: $todayDate)") - } else { - Log.d("UserPreferences", "이미 첫 알람 해제 ID가 저장되어 있음)") - } + suspend fun markFirstAlarmDismissedToday() { + dataStore.edit { pref -> + pref[Keys.FIRST_ALARM_DISMISSED_TODAY] = true + pref[Keys.FIRST_ALARM_DISMISSED_DATE_EPOCH] = todayEpoch() } } - suspend fun setOnboardingCompleted() { - dataStore.edit { preferences -> - preferences[Keys.ONBOARDING_COMPLETED] = true - } + suspend fun markUpdateNoticeDontShow(version: String) { + dataStore.edit { it[Keys.UPDATE_NOTICE_DONT_SHOW_VERSION] = version } } - suspend fun clearDismissedAlarmId() { - dataStore.edit { preferences -> - preferences.remove(Keys.FIRST_DISMISSED_ALARM_ID) - preferences.remove(Keys.DISMISSED_DATE) + suspend fun markUpdateNoticeShownToday() { + dataStore.edit { pref -> + pref[Keys.UPDATE_NOTICE_LAST_SHOWN_DATE_EPOCH] = todayEpoch() } } suspend fun clearUserData() { - dataStore.edit { preferences -> - preferences.clear() - } + dataStore.edit { it.clear() } } - suspend fun clearFortuneId() { - dataStore.edit { preferences -> - preferences.remove(Keys.FORTUNE_ID) - preferences.remove(Keys.FORTUNE_DATE) - preferences.remove(Keys.FORTUNE_IMAGE_ID) - preferences.remove(Keys.FORTUNE_SCORE) - preferences.remove(Keys.FORTUNE_CHECKED) + suspend fun clearFortuneData() { + dataStore.edit { pref -> + pref.remove(Keys.FORTUNE_ID) + pref.remove(Keys.FORTUNE_DATE_EPOCH) + pref.remove(Keys.FORTUNE_IMAGE_ID) + pref.remove(Keys.FORTUNE_SCORE) + pref.remove(Keys.FORTUNE_SEEN) + pref.remove(Keys.FORTUNE_TOOLTIP_SHOWN) + pref.remove(Keys.FORTUNE_CREATING) + pref.remove(Keys.FORTUNE_FAILED) } } } diff --git a/core/designsystem/src/main/res/drawable-xhdpi/ic_100_buble.png b/core/designsystem/src/main/res/drawable-xhdpi/ic_100_buble.png deleted file mode 100644 index d30beec6..00000000 Binary files a/core/designsystem/src/main/res/drawable-xhdpi/ic_100_buble.png and /dev/null differ diff --git a/core/designsystem/src/main/res/drawable-xhdpi/ic_fortune_delivering_speech_bubble.png b/core/designsystem/src/main/res/drawable-xhdpi/ic_fortune_delivering_speech_bubble.png new file mode 100644 index 00000000..c75d2db3 Binary files /dev/null and b/core/designsystem/src/main/res/drawable-xhdpi/ic_fortune_delivering_speech_bubble.png differ diff --git a/core/designsystem/src/main/res/drawable-xhdpi/ic_fortune_waiting_speech_bubble.png b/core/designsystem/src/main/res/drawable-xhdpi/ic_fortune_waiting_speech_bubble.png new file mode 100644 index 00000000..a119387d Binary files /dev/null and b/core/designsystem/src/main/res/drawable-xhdpi/ic_fortune_waiting_speech_bubble.png differ diff --git a/core/designsystem/src/main/res/drawable-xxhdpi/ic_100_buble.png b/core/designsystem/src/main/res/drawable-xxhdpi/ic_100_buble.png deleted file mode 100644 index 469a7b3f..00000000 Binary files a/core/designsystem/src/main/res/drawable-xxhdpi/ic_100_buble.png and /dev/null differ diff --git a/core/designsystem/src/main/res/drawable-xxhdpi/ic_fortune_delivering_speech_bubble.png b/core/designsystem/src/main/res/drawable-xxhdpi/ic_fortune_delivering_speech_bubble.png new file mode 100644 index 00000000..318d11d6 Binary files /dev/null and b/core/designsystem/src/main/res/drawable-xxhdpi/ic_fortune_delivering_speech_bubble.png differ diff --git a/core/designsystem/src/main/res/drawable-xxhdpi/ic_fortune_waiting_speech_bubble.png b/core/designsystem/src/main/res/drawable-xxhdpi/ic_fortune_waiting_speech_bubble.png new file mode 100644 index 00000000..83b665f4 Binary files /dev/null and b/core/designsystem/src/main/res/drawable-xxhdpi/ic_fortune_waiting_speech_bubble.png differ diff --git a/core/designsystem/src/main/res/drawable/ic_100_buble.png b/core/designsystem/src/main/res/drawable/ic_100_buble.png deleted file mode 100644 index e69978c4..00000000 Binary files a/core/designsystem/src/main/res/drawable/ic_100_buble.png and /dev/null differ diff --git a/core/designsystem/src/main/res/raw/fortune_loading.json b/core/designsystem/src/main/res/raw/fortune_loading.json new file mode 100644 index 00000000..cb570812 --- /dev/null +++ b/core/designsystem/src/main/res/raw/fortune_loading.json @@ -0,0 +1 @@ +{"nm":"컴포지션 2","h":500,"w":700,"meta":{"g":"@lottiefiles/toolkit-js 0.66.4","tc":"#202f44"},"layers":[{"ty":2,"nm":"Group 1948760243.png","sr":1,"st":1,"op":45,"ip":0,"ln":"2856","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[151.5,137]},"s":{"a":0,"k":[61.6,61.6,101.65]},"p":{"a":1,"k":[{"o":{"x":0,"y":0},"i":{"x":1,"y":1},"s":[130.5,159,0],"t":0},{"o":{"x":0,"y":0},"i":{"x":1,"y":1},"s":[167.139,178.955,0],"t":15},{"o":{"x":0,"y":0},"i":{"x":1,"y":1},"s":[233.8,193.529,0],"t":31},{"s":[285.485,192.267,0],"t":45}]},"r":{"a":1,"k":[{"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[17.4],"t":0},{"s":[32.4],"t":45}]},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"refId":"1","ind":1},{"ty":2,"nm":"Group 1948760248-1.png","sr":1,"st":0,"op":45,"ip":0,"ln":"2855","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[22,121.064]},"s":{"a":0,"k":[70.27,70.27,89.655]},"p":{"a":0,"k":[386,148,0]},"r":{"a":1,"k":[{"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[-9.304],"t":0},{"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[0],"t":45},{"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[10],"t":48},{"s":[0],"t":51}]},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"refId":"2","ind":2},{"ty":2,"nm":"Vector 27856.png","sr":1,"st":0,"op":45,"ip":0,"ln":"2854","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[93.94,9.052]},"s":{"a":0,"k":[75.628,72.572,100]},"p":{"a":0,"k":[310.5,242.5,0]},"r":{"a":0,"k":2},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"refId":"3","ind":3},{"ty":2,"nm":"Group 1948760248.png","sr":1,"st":0,"op":45,"ip":0,"ln":"2853","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[132.5,232]},"s":{"a":0,"k":[76.4,76.4,76.4]},"p":{"a":0,"k":[370,256,0]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"refId":"4","ind":4}],"v":"5.7.0","fr":30,"op":45,"ip":0,"assets":[{"id":"1","e":1,"w":303,"h":274,"p":"","u":""},{"id":"2","e":1,"w":100,"h":148,"p":"","u":""},{"id":"3","e":1,"w":154,"h":124,"p":"","u":""},{"id":"4","e":1,"w":265,"h":464,"p":"","u":""}]} \ No newline at end of file diff --git a/core/network/src/main/java/com/yapp/network/di/NetworkModule.kt b/core/network/src/main/java/com/yapp/network/di/NetworkModule.kt index 10276dfb..25dfb118 100644 --- a/core/network/src/main/java/com/yapp/network/di/NetworkModule.kt +++ b/core/network/src/main/java/com/yapp/network/di/NetworkModule.kt @@ -6,7 +6,6 @@ import dagger.Provides import dagger.hilt.InstallIn import dagger.hilt.components.SingletonComponent import kotlinx.serialization.json.Json -import okhttp3.Interceptor import okhttp3.MediaType.Companion.toMediaType import okhttp3.OkHttpClient import okhttp3.logging.HttpLoggingInterceptor @@ -39,21 +38,7 @@ object NetworkModule { @Provides @Singleton - @Auth - fun provideAuthOkHttpClient( - loggingInterceptor: HttpLoggingInterceptor, - authInterceptor: Interceptor, - ): OkHttpClient = - OkHttpClient.Builder() - .retryOnConnectionFailure(true) - .addInterceptor(loggingInterceptor) - .addInterceptor(authInterceptor) - .build() - - @Provides - @Singleton - @NoneAuth - fun provideNoneAuthOkHttpClient( + fun provideHttpClient( loggingInterceptor: HttpLoggingInterceptor, ): OkHttpClient = OkHttpClient.Builder() @@ -66,18 +51,8 @@ object NetworkModule { @Provides @Singleton - @Auth - fun provideAuthRetrofit(@Auth okHttpClient: OkHttpClient, buildConfigFieldProvider: BuildConfigFieldProvider): Retrofit = Retrofit.Builder() - .addConverterFactory(Json.asConverterFactory("application/json".toMediaType())) - .baseUrl(buildConfigFieldProvider.get().baseUrl) - .client(okHttpClient) - .build() - - @Provides - @Singleton - @NoneAuth - fun provideNoneAuthRetrofit( - @NoneAuth okHttpClient: OkHttpClient, + fun provideRetrofit( + okHttpClient: OkHttpClient, buildConfigFieldProvider: BuildConfigFieldProvider, json: Json, ): Retrofit = @@ -86,14 +61,4 @@ object NetworkModule { .baseUrl(buildConfigFieldProvider.get().baseUrl) .client(okHttpClient) .build() - - @Provides - @Singleton - @S3 - fun provideS3Retrofit(@NoneAuth okHttpClient: OkHttpClient, buildConfigFieldProvider: BuildConfigFieldProvider): Retrofit = - Retrofit.Builder() - .addConverterFactory(Json.asConverterFactory("application/json".toMediaType())) - .baseUrl(buildConfigFieldProvider.get().baseUrl) - .client(okHttpClient) - .build() } diff --git a/core/network/src/main/java/com/yapp/network/di/Qualifier.kt b/core/network/src/main/java/com/yapp/network/di/Qualifier.kt deleted file mode 100644 index 68817100..00000000 --- a/core/network/src/main/java/com/yapp/network/di/Qualifier.kt +++ /dev/null @@ -1,15 +0,0 @@ -package com.yapp.network.di - -import javax.inject.Qualifier - -@Qualifier -@Retention(AnnotationRetention.BINARY) -annotation class NoneAuth - -@Qualifier -@Retention(AnnotationRetention.BINARY) -annotation class Auth - -@Qualifier -@Retention(AnnotationRetention.BINARY) -annotation class S3 diff --git a/core/ui/src/main/java/com/yapp/ui/component/navigation/NavigationBarScrim.kt b/core/ui/src/main/java/com/yapp/ui/component/navigation/NavigationBarScrim.kt new file mode 100644 index 00000000..c0bcb5a7 --- /dev/null +++ b/core/ui/src/main/java/com/yapp/ui/component/navigation/NavigationBarScrim.kt @@ -0,0 +1,26 @@ +package com.yapp.ui.component.navigation + +import androidx.compose.foundation.background +import androidx.compose.foundation.layout.Box +import androidx.compose.foundation.layout.BoxScope +import androidx.compose.foundation.layout.WindowInsets +import androidx.compose.foundation.layout.fillMaxWidth +import androidx.compose.foundation.layout.navigationBars +import androidx.compose.foundation.layout.windowInsetsBottomHeight +import androidx.compose.runtime.Composable +import androidx.compose.ui.Alignment +import androidx.compose.ui.Modifier +import androidx.compose.ui.graphics.Color +import androidx.compose.ui.zIndex + +@Composable +fun BoxScope.NavigationBarScrim() { + Box( + modifier = Modifier + .align(Alignment.BottomCenter) + .fillMaxWidth() + .windowInsetsBottomHeight(WindowInsets.navigationBars) + .background(Color.Black) + .zIndex(1f), + ) +} diff --git a/data/src/main/java/com/yapp/data/local/datasource/AlarmLocalDataSource.kt b/data/src/main/java/com/yapp/data/local/datasource/AlarmLocalDataSource.kt index eb9fb350..8ff592d2 100644 --- a/data/src/main/java/com/yapp/data/local/datasource/AlarmLocalDataSource.kt +++ b/data/src/main/java/com/yapp/data/local/datasource/AlarmLocalDataSource.kt @@ -5,8 +5,6 @@ import com.yapp.domain.model.Alarm import kotlinx.coroutines.flow.Flow interface AlarmLocalDataSource { - val firstDismissedAlarmIdFlow: Flow - fun getAllAlarms(): Flow> fun getAlarmsByTime(hour: Int, minute: Int): Flow> suspend fun insertAlarm(alarm: AlarmEntity): Long @@ -14,6 +12,4 @@ interface AlarmLocalDataSource { suspend fun updateAlarmActive(id: Long, active: Boolean): Int suspend fun getAlarm(id: Long): Alarm? suspend fun deleteAlarm(id: Long): Int - suspend fun saveFirstDismissedAlarmId(alarmId: Long) - suspend fun clearDismissedAlarmId() } diff --git a/data/src/main/java/com/yapp/data/local/datasource/AlarmLocalDataSourceImpl.kt b/data/src/main/java/com/yapp/data/local/datasource/AlarmLocalDataSourceImpl.kt index 7c7425b2..03fecd55 100644 --- a/data/src/main/java/com/yapp/data/local/datasource/AlarmLocalDataSourceImpl.kt +++ b/data/src/main/java/com/yapp/data/local/datasource/AlarmLocalDataSourceImpl.kt @@ -3,7 +3,6 @@ package com.yapp.data.local.datasource import com.yapp.database.AlarmDao import com.yapp.database.AlarmEntity import com.yapp.database.toDomain -import com.yapp.datastore.UserPreferences import com.yapp.domain.model.Alarm import kotlinx.coroutines.flow.Flow import kotlinx.coroutines.flow.map @@ -11,10 +10,7 @@ import javax.inject.Inject class AlarmLocalDataSourceImpl @Inject constructor( private val alarmDao: AlarmDao, - private val userPreferences: UserPreferences, ) : AlarmLocalDataSource { - override val firstDismissedAlarmIdFlow: Flow = userPreferences.firstDismissedAlarmIdFlow - override fun getAllAlarms(): Flow> { return alarmDao.getAllAlarms() .map { alarmEntities -> alarmEntities.map { it.toDomain() } } @@ -45,12 +41,4 @@ class AlarmLocalDataSourceImpl @Inject constructor( override suspend fun deleteAlarm(id: Long): Int { return alarmDao.deleteAlarm(id) } - - override suspend fun saveFirstDismissedAlarmId(alarmId: Long) { - userPreferences.saveFirstDismissedAlarmId(alarmId) - } - - override suspend fun clearDismissedAlarmId() { - userPreferences.clearDismissedAlarmId() - } } diff --git a/data/src/main/java/com/yapp/data/local/datasource/FortuneLocalDataSource.kt b/data/src/main/java/com/yapp/data/local/datasource/FortuneLocalDataSource.kt index 149234f7..4e519ddb 100644 --- a/data/src/main/java/com/yapp/data/local/datasource/FortuneLocalDataSource.kt +++ b/data/src/main/java/com/yapp/data/local/datasource/FortuneLocalDataSource.kt @@ -1,20 +1,27 @@ package com.yapp.data.local.datasource +import com.yapp.domain.model.FortuneCreateStatus import kotlinx.coroutines.flow.Flow interface FortuneLocalDataSource { val fortuneIdFlow: Flow - val fortuneDateFlow: Flow + val fortuneDateEpochFlow: Flow val fortuneImageIdFlow: Flow val fortuneScoreFlow: Flow - val hasNewFortuneFlow: Flow - val firstDismissedAlarmIdFlow: Flow + val hasUnseenFortuneFlow: Flow + val shouldShowFortuneToolTipFlow: Flow + val isFirstAlarmDismissedTodayFlow: Flow - suspend fun saveFortuneId(fortuneId: Long) - suspend fun markFortuneAsChecked() + val fortuneCreateStatusFlow: Flow + + suspend fun markFortuneCreating() + suspend fun markFortuneCreated(fortuneId: Long) + suspend fun markFortuneFailed() + suspend fun markFortuneSeen() + suspend fun markFortuneTooltipShown() suspend fun saveFortuneImageId(imageResId: Int) suspend fun saveFortuneScore(score: Int) - suspend fun saveFirstDismissedAlarmId(alarmId: Long) - suspend fun clearDismissedAlarmId() - suspend fun clearFortuneId() + suspend fun markFirstAlarmDismissedToday() + + suspend fun clearFortuneData() } diff --git a/data/src/main/java/com/yapp/data/local/datasource/FortuneLocalDataSourceImpl.kt b/data/src/main/java/com/yapp/data/local/datasource/FortuneLocalDataSourceImpl.kt index 6dbe10f9..b8ab799f 100644 --- a/data/src/main/java/com/yapp/data/local/datasource/FortuneLocalDataSourceImpl.kt +++ b/data/src/main/java/com/yapp/data/local/datasource/FortuneLocalDataSourceImpl.kt @@ -1,6 +1,10 @@ package com.yapp.data.local.datasource import com.yapp.datastore.UserPreferences +import com.yapp.domain.model.FortuneCreateStatus +import kotlinx.coroutines.flow.combine +import kotlinx.coroutines.flow.distinctUntilChanged +import java.time.LocalDate import javax.inject.Inject class FortuneLocalDataSourceImpl @Inject constructor( @@ -8,18 +12,47 @@ class FortuneLocalDataSourceImpl @Inject constructor( ) : FortuneLocalDataSource { override val fortuneIdFlow = userPreferences.fortuneIdFlow - override val fortuneDateFlow = userPreferences.fortuneDateFlow + override val fortuneDateEpochFlow = userPreferences.fortuneDateEpochFlow override val fortuneImageIdFlow = userPreferences.fortuneImageIdFlow override val fortuneScoreFlow = userPreferences.fortuneScoreFlow - override val hasNewFortuneFlow = userPreferences.hasNewFortuneFlow - override val firstDismissedAlarmIdFlow = userPreferences.firstDismissedAlarmIdFlow + override val hasUnseenFortuneFlow = userPreferences.hasUnseenFortuneFlow + override val shouldShowFortuneToolTipFlow = userPreferences.shouldShowFortuneToolTipFlow + override val isFirstAlarmDismissedTodayFlow = userPreferences.isFirstAlarmDismissedTodayFlow - override suspend fun saveFortuneId(fortuneId: Long) { - userPreferences.saveFortuneId(fortuneId) + override val fortuneCreateStatusFlow = combine( + userPreferences.fortuneIdFlow, + userPreferences.fortuneDateEpochFlow, + userPreferences.isFortuneCreatingFlow, + userPreferences.isFortuneFailedFlow, + ) { fortuneId, fortuneDate, isCreating, isFailed -> + when { + isFailed -> FortuneCreateStatus.Failure + isCreating -> FortuneCreateStatus.Creating + fortuneId != null && fortuneDate == todayEpoch() -> FortuneCreateStatus.Success(fortuneId) + else -> FortuneCreateStatus.Idle + } + }.distinctUntilChanged() + + private fun todayEpoch(): Long = LocalDate.now().toEpochDay() + + override suspend fun markFortuneCreating() { + userPreferences.markFortuneCreating() + } + + override suspend fun markFortuneCreated(fortuneId: Long) { + userPreferences.markFortuneCreated(fortuneId) + } + + override suspend fun markFortuneFailed() { + userPreferences.markFortuneFailed() + } + + override suspend fun markFortuneSeen() { + userPreferences.markFortuneSeen() } - override suspend fun markFortuneAsChecked() { - userPreferences.markFortuneAsChecked() + override suspend fun markFortuneTooltipShown() { + userPreferences.markFortuneTooltipShown() } override suspend fun saveFortuneImageId(imageResId: Int) { @@ -30,15 +63,11 @@ class FortuneLocalDataSourceImpl @Inject constructor( userPreferences.saveFortuneScore(score) } - override suspend fun saveFirstDismissedAlarmId(alarmId: Long) { - userPreferences.saveFirstDismissedAlarmId(alarmId) - } - - override suspend fun clearDismissedAlarmId() { - userPreferences.clearDismissedAlarmId() + override suspend fun markFirstAlarmDismissedToday() { + userPreferences.markFirstAlarmDismissedToday() } - override suspend fun clearFortuneId() { - userPreferences.clearFortuneId() + override suspend fun clearFortuneData() { + userPreferences.clearFortuneData() } } diff --git a/data/src/main/java/com/yapp/data/local/datasource/UserLocalDataSource.kt b/data/src/main/java/com/yapp/data/local/datasource/UserLocalDataSource.kt index 37b4fc5a..3ad851df 100644 --- a/data/src/main/java/com/yapp/data/local/datasource/UserLocalDataSource.kt +++ b/data/src/main/java/com/yapp/data/local/datasource/UserLocalDataSource.kt @@ -6,9 +6,13 @@ interface UserLocalDataSource { val userIdFlow: Flow val userNameFlow: Flow val onboardingCompletedFlow: Flow + val updateNoticeDontShowVersionFlow: Flow + val updateNoticeLastShownDateEpochFlow: Flow suspend fun saveUserId(userId: Long) suspend fun saveUserName(userName: String) suspend fun setOnboardingCompleted() + suspend fun markUpdateNoticeDontShow(version: String) + suspend fun markUpdateNoticeShownToday() suspend fun clearUserData() } diff --git a/data/src/main/java/com/yapp/data/local/datasource/UserLocalDataSourceImpl.kt b/data/src/main/java/com/yapp/data/local/datasource/UserLocalDataSourceImpl.kt index 7e7d4324..187a7a59 100644 --- a/data/src/main/java/com/yapp/data/local/datasource/UserLocalDataSourceImpl.kt +++ b/data/src/main/java/com/yapp/data/local/datasource/UserLocalDataSourceImpl.kt @@ -11,6 +11,8 @@ class UserLocalDataSourceImpl @Inject constructor( override val userIdFlow: Flow = userPreferences.userIdFlow override val userNameFlow: Flow = userPreferences.userNameFlow override val onboardingCompletedFlow: Flow = userPreferences.onboardingCompletedFlow + override val updateNoticeDontShowVersionFlow: Flow = userPreferences.updateNoticeDontShowVersionFlow + override val updateNoticeLastShownDateEpochFlow: Flow = userPreferences.updateNoticeLastShownDateEpochFlow override suspend fun saveUserId(userId: Long) { userPreferences.saveUserId(userId) @@ -24,6 +26,14 @@ class UserLocalDataSourceImpl @Inject constructor( userPreferences.setOnboardingCompleted() } + override suspend fun markUpdateNoticeDontShow(version: String) { + userPreferences.markUpdateNoticeDontShow(version) + } + + override suspend fun markUpdateNoticeShownToday() { + userPreferences.markUpdateNoticeShownToday() + } + override suspend fun clearUserData() { userPreferences.clearUserData() } diff --git a/data/src/main/java/com/yapp/data/remote/di/ServiceModule.kt b/data/src/main/java/com/yapp/data/remote/di/ServiceModule.kt index 458db14b..be0e97f5 100644 --- a/data/src/main/java/com/yapp/data/remote/di/ServiceModule.kt +++ b/data/src/main/java/com/yapp/data/remote/di/ServiceModule.kt @@ -1,7 +1,6 @@ package com.yapp.data.remote.di import com.yapp.data.remote.service.ApiService -import com.yapp.network.di.NoneAuth import dagger.Module import dagger.Provides import dagger.hilt.InstallIn @@ -14,6 +13,6 @@ import javax.inject.Singleton object ServiceModule { @Provides @Singleton - fun providesApiService(@NoneAuth retrofit: Retrofit): ApiService = + fun providesApiService(retrofit: Retrofit): ApiService = retrofit.create(ApiService::class.java) } diff --git a/data/src/main/java/com/yapp/data/repositoryimpl/AlarmRepositoryImpl.kt b/data/src/main/java/com/yapp/data/repositoryimpl/AlarmRepositoryImpl.kt index a1c72135..50385b4f 100644 --- a/data/src/main/java/com/yapp/data/repositoryimpl/AlarmRepositoryImpl.kt +++ b/data/src/main/java/com/yapp/data/repositoryimpl/AlarmRepositoryImpl.kt @@ -16,8 +16,6 @@ class AlarmRepositoryImpl @Inject constructor( private val ringtoneManagerHelper: RingtoneManagerHelper, private val soundPlayer: SoundPlayer, ) : AlarmRepository { - override val firstDismissedAlarmIdFlow: Flow = alarmLocalDataSource.firstDismissedAlarmIdFlow - override suspend fun getAlarmSounds(): Result> = runCatching { ringtoneManagerHelper.getAlarmSounds().map { (title, uri) -> AlarmSound(title, uri) @@ -93,12 +91,4 @@ class AlarmRepositoryImpl @Inject constructor( throw Exception("No rows deleted") } } - - override suspend fun saveFirstDismissedAlarmId(alarmId: Long) { - alarmLocalDataSource.saveFirstDismissedAlarmId(alarmId) - } - - override suspend fun clearDismissedAlarmId() { - alarmLocalDataSource.clearDismissedAlarmId() - } } diff --git a/data/src/main/java/com/yapp/data/repositoryimpl/FortuneRepositoryImpl.kt b/data/src/main/java/com/yapp/data/repositoryimpl/FortuneRepositoryImpl.kt index d3abb0e9..1c761ba6 100644 --- a/data/src/main/java/com/yapp/data/repositoryimpl/FortuneRepositoryImpl.kt +++ b/data/src/main/java/com/yapp/data/repositoryimpl/FortuneRepositoryImpl.kt @@ -4,6 +4,7 @@ import com.yapp.data.local.datasource.FortuneLocalDataSource import com.yapp.data.remote.datasource.FortuneDataSource import com.yapp.data.remote.dto.response.toDomain import com.yapp.domain.model.Fortune +import com.yapp.domain.model.FortuneCreateStatus import com.yapp.domain.repository.FortuneRepository import kotlinx.coroutines.flow.Flow import javax.inject.Inject @@ -12,32 +13,35 @@ class FortuneRepositoryImpl @Inject constructor( private val fortuneLocalDataSource: FortuneLocalDataSource, private val fortuneRemoteDataSource: FortuneDataSource, ) : FortuneRepository { + override val fortuneIdFlow: Flow = fortuneLocalDataSource.fortuneIdFlow - override val fortuneDateFlow: Flow = fortuneLocalDataSource.fortuneDateFlow + override val fortuneDateEpochFlow: Flow = fortuneLocalDataSource.fortuneDateEpochFlow override val fortuneImageIdFlow: Flow = fortuneLocalDataSource.fortuneImageIdFlow override val fortuneScoreFlow: Flow = fortuneLocalDataSource.fortuneScoreFlow - override val hasNewFortuneFlow: Flow = fortuneLocalDataSource.hasNewFortuneFlow - override val firstDismissedAlarmIdFlow: Flow = fortuneLocalDataSource.firstDismissedAlarmIdFlow + override val hasUnseenFortuneFlow: Flow = fortuneLocalDataSource.hasUnseenFortuneFlow + override val shouldShowFortuneToolTipFlow: Flow = fortuneLocalDataSource.shouldShowFortuneToolTipFlow + override val isFirstAlarmDismissedTodayFlow: Flow = fortuneLocalDataSource.isFirstAlarmDismissedTodayFlow + + override val fortuneCreateStatusFlow: Flow = fortuneLocalDataSource.fortuneCreateStatusFlow - override suspend fun saveFortuneId(fortuneId: Long) = fortuneLocalDataSource.saveFortuneId(fortuneId) - override suspend fun markFortuneAsChecked() = fortuneLocalDataSource.markFortuneAsChecked() + override suspend fun markFortuneAsCreating() = fortuneLocalDataSource.markFortuneCreating() + override suspend fun markFortuneAsCreated(fortuneId: Long) = fortuneLocalDataSource.markFortuneCreated(fortuneId) + override suspend fun markFortuneAsFailed() = fortuneLocalDataSource.markFortuneFailed() + override suspend fun markFortuneSeen() = fortuneLocalDataSource.markFortuneSeen() + override suspend fun markFortuneTooltipShown() = fortuneLocalDataSource.markFortuneTooltipShown() override suspend fun saveFortuneImageId(imageResId: Int) = fortuneLocalDataSource.saveFortuneImageId(imageResId) override suspend fun saveFortuneScore(score: Int) = fortuneLocalDataSource.saveFortuneScore(score) - override suspend fun saveFirstDismissedAlarmId(alarmId: Long) = fortuneLocalDataSource.saveFirstDismissedAlarmId(alarmId) - override suspend fun clearDismissedAlarmId() = fortuneLocalDataSource.clearDismissedAlarmId() - override suspend fun clearFortuneId() = fortuneLocalDataSource.clearFortuneId() + override suspend fun markFirstAlarmDismissedToday() = fortuneLocalDataSource.markFirstAlarmDismissedToday() + + override suspend fun clearFortuneData() = fortuneLocalDataSource.clearFortuneData() override suspend fun postFortune(userId: Long): Result { return fortuneRemoteDataSource.postFortune(userId) - .mapCatching { fortuneResponse -> - fortuneResponse.toDomain() - } + .mapCatching { it.toDomain() } } override suspend fun getFortune(fortuneId: Long): Result { return fortuneRemoteDataSource.getFortune(fortuneId) - .mapCatching { fortuneResponse -> - fortuneResponse.toDomain() - } + .mapCatching { it.toDomain() } } } diff --git a/data/src/main/java/com/yapp/data/repositoryimpl/UserInfoRepositoryImpl.kt b/data/src/main/java/com/yapp/data/repositoryimpl/UserInfoRepositoryImpl.kt index d96ca6be..818e232d 100644 --- a/data/src/main/java/com/yapp/data/repositoryimpl/UserInfoRepositoryImpl.kt +++ b/data/src/main/java/com/yapp/data/repositoryimpl/UserInfoRepositoryImpl.kt @@ -17,10 +17,14 @@ class UserInfoRepositoryImpl @Inject constructor( override val userIdFlow: Flow = userLocalDataSource.userIdFlow override val userNameFlow: Flow = userLocalDataSource.userNameFlow override val onboardingCompletedFlow: Flow = userLocalDataSource.onboardingCompletedFlow + override val updateNoticeDontShowVersionFlow: Flow = userLocalDataSource.updateNoticeDontShowVersionFlow + override val updateNoticeLastShownDateEpochFlow: Flow = userLocalDataSource.updateNoticeLastShownDateEpochFlow override suspend fun saveUserId(userId: Long) = userLocalDataSource.saveUserId(userId) override suspend fun saveUserName(userName: String) = userLocalDataSource.saveUserName(userName) override suspend fun setOnboardingCompleted() = userLocalDataSource.setOnboardingCompleted() + override suspend fun markUpdateNoticeDontShow(version: String) = userLocalDataSource.markUpdateNoticeDontShow(version) + override suspend fun markUpdateNoticeShownToday() = userLocalDataSource.markUpdateNoticeShownToday() override suspend fun clearUserData() = userLocalDataSource.clearUserData() override suspend fun getUserInfo(userId: Long): Result { diff --git a/domain/src/main/java/com/yapp/domain/model/AlarmDay.kt b/domain/src/main/java/com/yapp/domain/model/AlarmDay.kt index beaead69..7f349ed5 100644 --- a/domain/src/main/java/com/yapp/domain/model/AlarmDay.kt +++ b/domain/src/main/java/com/yapp/domain/model/AlarmDay.kt @@ -1,5 +1,7 @@ package com.yapp.domain.model +import java.time.DayOfWeek + enum class AlarmDay(val bitValue: Int) { SUN(0b0000001), // 1 MON(0b0000010), // 2 @@ -11,8 +13,13 @@ enum class AlarmDay(val bitValue: Int) { ; } -fun AlarmDay.toDayOfWeek(): java.time.DayOfWeek { - return java.time.DayOfWeek.of(((this.ordinal + 6) % 7) + 1) +fun AlarmDay.toDayOfWeek(): DayOfWeek { + return DayOfWeek.of(((this.ordinal + 6) % 7) + 1) +} + +fun DayOfWeek.toAlarmDay(): AlarmDay { + val index = (this.value % 7) + return AlarmDay.entries[index] } fun Set.toRepeatDays(): Int { diff --git a/domain/src/main/java/com/yapp/domain/model/FortuneCreateStatus.kt b/domain/src/main/java/com/yapp/domain/model/FortuneCreateStatus.kt new file mode 100644 index 00000000..27ae9ad0 --- /dev/null +++ b/domain/src/main/java/com/yapp/domain/model/FortuneCreateStatus.kt @@ -0,0 +1,8 @@ +package com.yapp.domain.model + +sealed class FortuneCreateStatus { + data object Idle : FortuneCreateStatus() + data object Creating : FortuneCreateStatus() + data class Success(val fortuneId: Long) : FortuneCreateStatus() + data object Failure : FortuneCreateStatus() +} diff --git a/domain/src/main/java/com/yapp/domain/model/MissionMode.kt b/domain/src/main/java/com/yapp/domain/model/MissionMode.kt new file mode 100644 index 00000000..16009fc2 --- /dev/null +++ b/domain/src/main/java/com/yapp/domain/model/MissionMode.kt @@ -0,0 +1,13 @@ +package com.yapp.domain.model + +enum class MissionMode { + REAL, + PREVIEW, + ; + + companion object { + fun fromRaw(raw: String?): MissionMode { + return raw?.let { entries.find { it.name == raw } } ?: REAL + } + } +} diff --git a/domain/src/main/java/com/yapp/domain/repository/AlarmRepository.kt b/domain/src/main/java/com/yapp/domain/repository/AlarmRepository.kt index f7dac361..60123473 100644 --- a/domain/src/main/java/com/yapp/domain/repository/AlarmRepository.kt +++ b/domain/src/main/java/com/yapp/domain/repository/AlarmRepository.kt @@ -6,8 +6,6 @@ import com.yapp.domain.model.AlarmSound import kotlinx.coroutines.flow.Flow interface AlarmRepository { - val firstDismissedAlarmIdFlow: Flow - suspend fun getAlarmSounds(): Result> fun initializeSoundPlayer(uri: Uri) fun playAlarmSound(volume: Int) @@ -21,6 +19,4 @@ interface AlarmRepository { suspend fun updateAlarmActive(id: Long, active: Boolean): Result suspend fun getAlarm(id: Long): Result suspend fun deleteAlarm(id: Long): Result - suspend fun saveFirstDismissedAlarmId(alarmId: Long) - suspend fun clearDismissedAlarmId() } diff --git a/domain/src/main/java/com/yapp/domain/repository/FortuneRepository.kt b/domain/src/main/java/com/yapp/domain/repository/FortuneRepository.kt index 23598e3c..372fd5fe 100644 --- a/domain/src/main/java/com/yapp/domain/repository/FortuneRepository.kt +++ b/domain/src/main/java/com/yapp/domain/repository/FortuneRepository.kt @@ -1,23 +1,31 @@ package com.yapp.domain.repository import com.yapp.domain.model.Fortune +import com.yapp.domain.model.FortuneCreateStatus import kotlinx.coroutines.flow.Flow interface FortuneRepository { val fortuneIdFlow: Flow - val fortuneDateFlow: Flow + val fortuneDateEpochFlow: Flow val fortuneImageIdFlow: Flow val fortuneScoreFlow: Flow - val hasNewFortuneFlow: Flow - val firstDismissedAlarmIdFlow: Flow + val hasUnseenFortuneFlow: Flow + val shouldShowFortuneToolTipFlow: Flow + val isFirstAlarmDismissedTodayFlow: Flow - suspend fun saveFortuneId(fortuneId: Long) - suspend fun markFortuneAsChecked() + val fortuneCreateStatusFlow: Flow + + suspend fun markFortuneAsCreating() + suspend fun markFortuneAsCreated(fortuneId: Long) + suspend fun markFortuneAsFailed() + suspend fun markFortuneSeen() + suspend fun markFortuneTooltipShown() suspend fun saveFortuneImageId(imageResId: Int) suspend fun saveFortuneScore(score: Int) - suspend fun saveFirstDismissedAlarmId(alarmId: Long) - suspend fun clearDismissedAlarmId() - suspend fun clearFortuneId() + suspend fun markFirstAlarmDismissedToday() + + suspend fun clearFortuneData() + suspend fun postFortune(userId: Long): Result suspend fun getFortune(fortuneId: Long): Result } diff --git a/domain/src/main/java/com/yapp/domain/repository/UserInfoRepository.kt b/domain/src/main/java/com/yapp/domain/repository/UserInfoRepository.kt index bda28291..a9df412e 100644 --- a/domain/src/main/java/com/yapp/domain/repository/UserInfoRepository.kt +++ b/domain/src/main/java/com/yapp/domain/repository/UserInfoRepository.kt @@ -8,10 +8,14 @@ interface UserInfoRepository { val userIdFlow: Flow val userNameFlow: Flow val onboardingCompletedFlow: Flow + val updateNoticeDontShowVersionFlow: Flow + val updateNoticeLastShownDateEpochFlow: Flow suspend fun saveUserId(userId: Long) suspend fun saveUserName(userName: String) suspend fun setOnboardingCompleted() + suspend fun markUpdateNoticeDontShow(version: String) + suspend fun markUpdateNoticeShownToday() suspend fun clearUserData() suspend fun getUserInfo(userId: Long): Result diff --git a/feature/alarm-interaction/src/main/java/com/yapp/alarm/interaction/AlarmInteractionActivity.kt b/feature/alarm-interaction/src/main/java/com/yapp/alarm/interaction/AlarmInteractionActivity.kt index 92d52cee..e95a93b2 100644 --- a/feature/alarm-interaction/src/main/java/com/yapp/alarm/interaction/AlarmInteractionActivity.kt +++ b/feature/alarm-interaction/src/main/java/com/yapp/alarm/interaction/AlarmInteractionActivity.kt @@ -10,7 +10,10 @@ import android.os.Bundle import androidx.activity.ComponentActivity import androidx.activity.compose.setContent import androidx.activity.enableEdgeToEdge +import androidx.compose.foundation.layout.Box +import androidx.compose.foundation.layout.navigationBarsPadding import androidx.compose.runtime.DisposableEffect +import androidx.compose.ui.Modifier import androidx.core.util.Consumer import androidx.navigation.compose.NavHost import com.yapp.alarm.AlarmConstants @@ -18,6 +21,7 @@ import com.yapp.alarm.receivers.AlarmInteractionActivityReceiver import com.yapp.common.navigation.rememberOrbitNavigator import com.yapp.common.navigation.route.AlarmInteractionBaseRoute import com.yapp.domain.model.Alarm +import com.yapp.ui.component.navigation.NavigationBarScrim import dagger.hilt.android.AndroidEntryPoint @AndroidEntryPoint @@ -45,14 +49,19 @@ class AlarmInteractionActivity : ComponentActivity() { setContent { val navigator = rememberOrbitNavigator() - NavHost( - navController = navigator.navController, - startDestination = AlarmInteractionBaseRoute, - ) { - alarmInteractionNavGraph( - navigator = navigator, - alarm = alarm, - ) + Box { + NavHost( + modifier = Modifier.navigationBarsPadding(), + navController = navigator.navController, + startDestination = AlarmInteractionBaseRoute, + ) { + alarmInteractionNavGraph( + navigator = navigator, + alarm = alarm, + ) + } + + NavigationBarScrim() } DisposableEffect(this, navigator.navController) { diff --git a/feature/alarm-interaction/src/main/java/com/yapp/alarm/interaction/action/AlarmActionContract.kt b/feature/alarm-interaction/src/main/java/com/yapp/alarm/interaction/action/AlarmActionContract.kt index ea4d0b68..9eee8bad 100644 --- a/feature/alarm-interaction/src/main/java/com/yapp/alarm/interaction/action/AlarmActionContract.kt +++ b/feature/alarm-interaction/src/main/java/com/yapp/alarm/interaction/action/AlarmActionContract.kt @@ -14,7 +14,7 @@ class AlarmActionContract { val snoozeEnabled: Boolean = true, val snoozeInterval: Int = 5, val snoozeCount: Int = 5, - val isFirstMission: Boolean? = null, + val shouldShowMissionStart: Boolean? = null, ) : UiState sealed class Action { diff --git a/feature/alarm-interaction/src/main/java/com/yapp/alarm/interaction/action/AlarmActionScreen.kt b/feature/alarm-interaction/src/main/java/com/yapp/alarm/interaction/action/AlarmActionScreen.kt index d58bdda1..8ff13cb9 100644 --- a/feature/alarm-interaction/src/main/java/com/yapp/alarm/interaction/action/AlarmActionScreen.kt +++ b/feature/alarm-interaction/src/main/java/com/yapp/alarm/interaction/action/AlarmActionScreen.kt @@ -84,7 +84,7 @@ internal fun AlarmActionScreen( snoozeEnabled = state.snoozeEnabled, snoozeInterval = state.snoozeInterval, snoozeCount = state.snoozeCount, - isFirstMission = state.isFirstMission, + isFirstMission = state.shouldShowMissionStart, onSnoozeClick = { processAction(AlarmActionContract.Action.Snooze) }, onDismissClick = { processAction(AlarmActionContract.Action.Dismiss) diff --git a/feature/alarm-interaction/src/main/java/com/yapp/alarm/interaction/action/AlarmActionViewModel.kt b/feature/alarm-interaction/src/main/java/com/yapp/alarm/interaction/action/AlarmActionViewModel.kt index a15a85c5..49182b15 100644 --- a/feature/alarm-interaction/src/main/java/com/yapp/alarm/interaction/action/AlarmActionViewModel.kt +++ b/feature/alarm-interaction/src/main/java/com/yapp/alarm/interaction/action/AlarmActionViewModel.kt @@ -6,10 +6,9 @@ import androidx.lifecycle.ViewModel import com.yapp.alarm.pendingIntent.interaction.createAlarmDismissIntent import com.yapp.alarm.pendingIntent.interaction.createAlarmSnoozeIntent import com.yapp.domain.model.Alarm -import com.yapp.domain.repository.FortuneRepository +import com.yapp.domain.model.MissionType import dagger.hilt.android.lifecycle.HiltViewModel import kotlinx.coroutines.delay -import kotlinx.coroutines.flow.firstOrNull import org.orbitmvi.orbit.Container import org.orbitmvi.orbit.ContainerHost import org.orbitmvi.orbit.syntax.simple.intent @@ -18,7 +17,6 @@ import org.orbitmvi.orbit.syntax.simple.reduce import org.orbitmvi.orbit.viewmodel.container import java.time.LocalDate import java.time.LocalTime -import java.time.format.DateTimeFormatter import java.time.format.TextStyle import java.util.Locale import javax.inject.Inject @@ -26,14 +24,13 @@ import javax.inject.Inject @HiltViewModel class AlarmActionViewModel @Inject constructor( private val app: Application, - private val fortuneRepository: FortuneRepository, savedStateHandle: SavedStateHandle, ) : ViewModel(), ContainerHost { override val container: Container = container( initialState = AlarmActionContract.State(), ) { - fetchIsFirstMission() + fetchShouldShowMissionStart() initializeAlarmState() startClock() } @@ -58,13 +55,9 @@ class AlarmActionViewModel @Inject constructor( } } - private fun fetchIsFirstMission() = intent { - val fortuneDate = fortuneRepository.fortuneDateFlow.firstOrNull() - val todayDate = LocalDate.now().format(DateTimeFormatter.ISO_DATE) - val isFirstMission = fortuneDate != todayDate - + private fun fetchShouldShowMissionStart() = intent { reduce { - state.copy(isFirstMission = isFirstMission) + state.copy(shouldShowMissionStart = (alarm?.missionType ?: MissionType.NONE) != MissionType.NONE) } } diff --git a/feature/alarm-interaction/src/main/java/com/yapp/alarm/interaction/snooze/AlarmSnoozeTimerViewModel.kt b/feature/alarm-interaction/src/main/java/com/yapp/alarm/interaction/snooze/AlarmSnoozeTimerViewModel.kt index 1f0e6d1b..05b6c798 100644 --- a/feature/alarm-interaction/src/main/java/com/yapp/alarm/interaction/snooze/AlarmSnoozeTimerViewModel.kt +++ b/feature/alarm-interaction/src/main/java/com/yapp/alarm/interaction/snooze/AlarmSnoozeTimerViewModel.kt @@ -17,7 +17,6 @@ import org.orbitmvi.orbit.viewmodel.container import java.time.LocalDate import java.time.LocalDateTime import java.time.ZoneId -import java.time.format.DateTimeFormatter import javax.inject.Inject import kotlin.math.max @@ -45,8 +44,8 @@ class AlarmSnoozeTimerViewModel @Inject constructor( } private fun fetchIsFirstMission() = intent { - val fortuneDate = fortuneRepository.fortuneDateFlow.firstOrNull() - val todayDate = LocalDate.now().format(DateTimeFormatter.ISO_DATE) + val fortuneDate = fortuneRepository.fortuneDateEpochFlow.firstOrNull() + val todayDate = LocalDate.now().toEpochDay() val isFirstMission = fortuneDate != todayDate reduce { diff --git a/feature/fortune/build.gradle.kts b/feature/fortune/build.gradle.kts index ae450155..543510e8 100644 --- a/feature/fortune/build.gradle.kts +++ b/feature/fortune/build.gradle.kts @@ -12,10 +12,14 @@ dependencies { implementation(projects.core.ui) implementation(projects.core.common) implementation(projects.core.analytics) + implementation(projects.core.alarm) implementation(libs.orbit.core) implementation(libs.orbit.compose) implementation(libs.orbit.viewmodel) implementation(libs.coil.compose) + implementation(libs.androidx.work.runtime) + testImplementation(libs.androidx.work.testing) + androidTestImplementation(libs.androidx.work.testing) implementation(projects.domain) implementation(projects.core.media) } diff --git a/feature/fortune/src/main/java/com/yapp/fortune/FortuneNavGraph.kt b/feature/fortune/src/main/java/com/yapp/fortune/FortuneNavGraph.kt index 5f1b5133..0a7b62ce 100644 --- a/feature/fortune/src/main/java/com/yapp/fortune/FortuneNavGraph.kt +++ b/feature/fortune/src/main/java/com/yapp/fortune/FortuneNavGraph.kt @@ -5,6 +5,7 @@ import androidx.compose.runtime.LaunchedEffect import androidx.compose.runtime.rememberCoroutineScope import androidx.navigation.NavGraphBuilder import androidx.navigation.compose.composable +import androidx.navigation.navDeepLink import androidx.navigation.navOptions import androidx.navigation.navigation import com.yapp.common.navigation.OrbitNavigator @@ -19,7 +20,11 @@ fun NavGraphBuilder.fortuneNavGraph( snackBarHostState: SnackbarHostState, ) { navigation(startDestination = FortuneDestination.Fortune) { - composable { backStackEntry -> + composable( + deepLinks = listOf( + navDeepLink { uriPattern = "orbitapp://fortune" }, + ), + ) { backStackEntry -> val viewModel = backStackEntry.sharedHiltViewModel(navigator.navController) val coroutineScope = rememberCoroutineScope() diff --git a/feature/fortune/src/main/java/com/yapp/fortune/FortuneScreen.kt b/feature/fortune/src/main/java/com/yapp/fortune/FortuneScreen.kt index f5109a3b..1eb97abc 100644 --- a/feature/fortune/src/main/java/com/yapp/fortune/FortuneScreen.kt +++ b/feature/fortune/src/main/java/com/yapp/fortune/FortuneScreen.kt @@ -3,11 +3,13 @@ package com.yapp.fortune import androidx.activity.compose.BackHandler import androidx.compose.foundation.Image import androidx.compose.foundation.background +import androidx.compose.foundation.layout.Arrangement import androidx.compose.foundation.layout.Box import androidx.compose.foundation.layout.Column import androidx.compose.foundation.layout.fillMaxSize +import androidx.compose.foundation.layout.height import androidx.compose.foundation.layout.navigationBarsPadding -import androidx.compose.foundation.layout.size +import androidx.compose.foundation.layout.width import androidx.compose.foundation.pager.PagerState import androidx.compose.foundation.pager.rememberPagerState import androidx.compose.runtime.Composable @@ -15,6 +17,7 @@ import androidx.compose.runtime.LaunchedEffect import androidx.compose.runtime.getValue import androidx.compose.runtime.mutableIntStateOf import androidx.compose.runtime.mutableLongStateOf +import androidx.compose.runtime.mutableStateOf import androidx.compose.runtime.remember import androidx.compose.runtime.setValue import androidx.compose.ui.Alignment @@ -34,6 +37,7 @@ import com.yapp.fortune.component.FortuneTopAppBar import com.yapp.fortune.component.SlidingIndicator import com.yapp.fortune.page.FortunePager import com.yapp.ui.component.lottie.LottieAnimation +import kotlinx.coroutines.delay import java.math.BigDecimal import java.math.RoundingMode @@ -184,21 +188,49 @@ fun FortuneScreen( @Composable fun FortuneLoadingScreen() { + var isDelivering by remember { mutableStateOf(false) } + + LaunchedEffect(Unit) { + while (true) { + delay(2000) + isDelivering = !isDelivering + } + } + Box( modifier = Modifier .fillMaxSize() .background(OrbitTheme.colors.gray_900.copy(alpha = 0.7f)), contentAlignment = Alignment.Center, ) { - LottieAnimation( - modifier = Modifier - .size(70.dp), - resId = core.designsystem.R.raw.star_loading, - ) + Column( + horizontalAlignment = Alignment.CenterHorizontally, + verticalArrangement = Arrangement.spacedBy(6.dp), + ) { + val imageRes = if (isDelivering) { + core.designsystem.R.drawable.ic_fortune_delivering_speech_bubble + } else { + core.designsystem.R.drawable.ic_fortune_waiting_speech_bubble + } + Image( + painter = painterResource(id = imageRes), + contentDescription = null, + ) + + LottieAnimation( + modifier = Modifier + .width(375.dp) + .height(267.dp), + resId = core.designsystem.R.raw.fortune_loading, + ) + } } } @Composable @Preview -fun FortuneRoutePreview() { +private fun FortuneLoadingScreenPreview() { + OrbitTheme { + FortuneLoadingScreen() + } } diff --git a/feature/fortune/src/main/java/com/yapp/fortune/FortuneViewModel.kt b/feature/fortune/src/main/java/com/yapp/fortune/FortuneViewModel.kt index 034c1590..4a83a561 100644 --- a/feature/fortune/src/main/java/com/yapp/fortune/FortuneViewModel.kt +++ b/feature/fortune/src/main/java/com/yapp/fortune/FortuneViewModel.kt @@ -4,11 +4,13 @@ import android.app.Application import android.util.Log import androidx.annotation.DrawableRes import androidx.lifecycle.ViewModel +import com.yapp.domain.model.FortuneCreateStatus import com.yapp.domain.repository.FortuneRepository import com.yapp.fortune.page.toFortunePages import com.yapp.media.decoder.ImageUtils import com.yapp.media.storage.ImageSaver import dagger.hilt.android.lifecycle.HiltViewModel +import kotlinx.coroutines.flow.first import kotlinx.coroutines.flow.firstOrNull import org.orbitmvi.orbit.Container import org.orbitmvi.orbit.ContainerHost @@ -16,8 +18,6 @@ import org.orbitmvi.orbit.syntax.simple.intent import org.orbitmvi.orbit.syntax.simple.postSideEffect import org.orbitmvi.orbit.syntax.simple.reduce import org.orbitmvi.orbit.viewmodel.container -import java.time.LocalDate -import java.time.format.DateTimeFormatter import javax.inject.Inject @HiltViewModel @@ -30,7 +30,7 @@ class FortuneViewModel @Inject constructor( override val container: Container = container( initialState = FortuneContract.State(), ) { - loadFortune() + observeFortune() } fun processAction(action: FortuneContract.Action) { @@ -50,14 +50,31 @@ class FortuneViewModel @Inject constructor( } } - private fun loadFortune() = intent { - val fortuneId = fortuneRepository.fortuneIdFlow.firstOrNull() - val firstDismissedAlarmId = fortuneRepository.firstDismissedAlarmIdFlow.firstOrNull() - val fortuneDate = fortuneRepository.fortuneDateFlow.firstOrNull() - fortuneId?.let { fetchAndUpdateFortune(it, firstDismissedAlarmId, fortuneDate) } + private fun observeFortune() = intent { + fortuneRepository.fortuneCreateStatusFlow.collect { status -> + when (status) { + is FortuneCreateStatus.Creating -> { + reduce { state.copy(isLoading = true) } + } + + is FortuneCreateStatus.Success -> { + fetchAndUpdateFortune( + fortuneId = status.fortuneId, + isFirstAlarmDismissedToday = fortuneRepository.isFirstAlarmDismissedTodayFlow.first(), + ) + } + + is FortuneCreateStatus.Failure, FortuneCreateStatus.Idle -> { + postSideEffect(FortuneContract.SideEffect.NavigateToHome) + } + } + } } - private fun fetchAndUpdateFortune(fortuneId: Long, firstDismissedAlarmId: Long?, fortuneDate: String?) = intent { + private fun fetchAndUpdateFortune( + fortuneId: Long, + isFirstAlarmDismissedToday: Boolean, + ) = intent { reduce { state.copy(isLoading = true) } fortuneRepository.getFortune(fortuneId).onSuccess { fortune -> @@ -65,8 +82,9 @@ class FortuneViewModel @Inject constructor( val imageId = savedImageId ?: getRandomImage() val formattedTitle = fortune.dailyFortuneTitle.replace(",", ",\n").trim() - val todayDate = LocalDate.now().format(DateTimeFormatter.ISO_DATE) - val hasReward = (fortuneDate == todayDate) && (firstDismissedAlarmId != null) + + fortuneRepository.markFortuneSeen() + reduce { state.copy( isLoading = false, @@ -75,7 +93,7 @@ class FortuneViewModel @Inject constructor( avgFortuneScore = fortune.avgFortuneScore, fortunePages = fortune.toFortunePages(), fortuneImageId = imageId, - hasReward = hasReward, + hasReward = isFirstAlarmDismissedToday, ) } }.onFailure { error -> diff --git a/feature/fortune/src/main/java/com/yapp/fortune/di/SchedulerModule.kt b/feature/fortune/src/main/java/com/yapp/fortune/di/SchedulerModule.kt new file mode 100644 index 00000000..47fbd0b0 --- /dev/null +++ b/feature/fortune/src/main/java/com/yapp/fortune/di/SchedulerModule.kt @@ -0,0 +1,19 @@ +package com.yapp.fortune.di + +import com.yapp.alarm.scheduler.PostFortuneTaskScheduler +import com.yapp.fortune.scheduler.WorkManagerPostFortuneTaskScheduler +import dagger.Binds +import dagger.Module +import dagger.hilt.InstallIn +import dagger.hilt.components.SingletonComponent +import javax.inject.Singleton + +@Module +@InstallIn(SingletonComponent::class) +abstract class SchedulerModule { + @Binds + @Singleton + abstract fun bindsPostFortuneTaskScheduler( + postFortuneTaskScheduler: WorkManagerPostFortuneTaskScheduler, + ): PostFortuneTaskScheduler +} diff --git a/feature/fortune/src/main/java/com/yapp/fortune/page/FortunePager.kt b/feature/fortune/src/main/java/com/yapp/fortune/page/FortunePager.kt index 08b7cfc5..aa1dc78b 100644 --- a/feature/fortune/src/main/java/com/yapp/fortune/page/FortunePager.kt +++ b/feature/fortune/src/main/java/com/yapp/fortune/page/FortunePager.kt @@ -51,11 +51,13 @@ fun FortunePager( val index = (page - 1).coerceIn(0, state.fortunePages.lastIndex) FortunePageLayout(state.fortunePages[index]) } + 5 -> FortuneCompletePage( hasReward = state.hasReward, onCompleteClick = onNextStep, onNavigateToHome = onNavigateToHome, ) + else -> {} } } diff --git a/feature/fortune/src/main/java/com/yapp/fortune/scheduler/WorkManagerPostFortuneTaskScheduler.kt b/feature/fortune/src/main/java/com/yapp/fortune/scheduler/WorkManagerPostFortuneTaskScheduler.kt new file mode 100644 index 00000000..36e49c4f --- /dev/null +++ b/feature/fortune/src/main/java/com/yapp/fortune/scheduler/WorkManagerPostFortuneTaskScheduler.kt @@ -0,0 +1,32 @@ +package com.yapp.fortune.scheduler + +import android.content.Context +import androidx.work.BackoffPolicy +import androidx.work.Constraints +import androidx.work.ExistingWorkPolicy +import androidx.work.NetworkType +import androidx.work.OneTimeWorkRequestBuilder +import androidx.work.WorkManager +import com.yapp.alarm.scheduler.PostFortuneTaskScheduler +import com.yapp.fortune.worker.PostFortuneWorker +import dagger.hilt.android.qualifiers.ApplicationContext +import java.time.LocalDate +import java.util.concurrent.TimeUnit +import javax.inject.Inject + +class WorkManagerPostFortuneTaskScheduler @Inject constructor( + @ApplicationContext private val context: Context, +) : PostFortuneTaskScheduler { + override fun enqueueOnceForToday() { + val name = "post_fortune_${LocalDate.now()}" + val constraints = Constraints.Builder() + .setRequiredNetworkType(NetworkType.CONNECTED) + .build() + val req = OneTimeWorkRequestBuilder() + .setConstraints(constraints) + .setBackoffCriteria(BackoffPolicy.EXPONENTIAL, 15, TimeUnit.SECONDS) + .build() + WorkManager.getInstance(context) + .enqueueUniqueWork(name, ExistingWorkPolicy.KEEP, req) + } +} diff --git a/feature/fortune/src/main/java/com/yapp/fortune/worker/PostFortuneWorker.kt b/feature/fortune/src/main/java/com/yapp/fortune/worker/PostFortuneWorker.kt new file mode 100644 index 00000000..2dd14f72 --- /dev/null +++ b/feature/fortune/src/main/java/com/yapp/fortune/worker/PostFortuneWorker.kt @@ -0,0 +1,63 @@ +package com.yapp.fortune.worker + +import android.content.Context +import androidx.hilt.work.HiltWorker +import androidx.work.CoroutineWorker +import androidx.work.WorkerParameters +import com.yapp.domain.model.FortuneCreateStatus +import com.yapp.domain.repository.FortuneRepository +import com.yapp.domain.repository.UserInfoRepository +import dagger.assisted.Assisted +import dagger.assisted.AssistedInject +import kotlinx.coroutines.flow.first +import kotlinx.coroutines.flow.firstOrNull + +@HiltWorker +class PostFortuneWorker @AssistedInject constructor( + @Assisted appContext: Context, + @Assisted params: WorkerParameters, + private val fortuneRepository: FortuneRepository, + private val userInfoRepository: UserInfoRepository, +) : CoroutineWorker(appContext, params) { + + override suspend fun doWork(): Result { + when (fortuneRepository.fortuneCreateStatusFlow.first()) { + is FortuneCreateStatus.Creating, + is FortuneCreateStatus.Success, + -> { + return Result.success() + } + FortuneCreateStatus.Failure, + FortuneCreateStatus.Idle, + -> { + val userId = userInfoRepository.userIdFlow.firstOrNull() + ?: run { + // 사용자 없으면 실패 상태 표시 후 실패 반환 + fortuneRepository.markFortuneAsFailed() + return Result.failure() + } + + return try { + fortuneRepository.markFortuneAsCreating() + + val result = fortuneRepository.postFortune(userId) + result.fold( + onSuccess = { fortune -> + fortuneRepository.markFortuneAsCreated(fortune.id) + fortuneRepository.saveFortuneScore(fortune.avgFortuneScore) + Result.success() + }, + onFailure = { + fortuneRepository.markFortuneAsFailed() + // WM 백오프 규칙에 따라 재시도 + Result.retry() + }, + ) + } catch (_: Throwable) { + fortuneRepository.markFortuneAsFailed() + Result.retry() + } + } + } + } +} diff --git a/feature/home/build.gradle.kts b/feature/home/build.gradle.kts index 9ef0f667..0b90300c 100644 --- a/feature/home/build.gradle.kts +++ b/feature/home/build.gradle.kts @@ -19,4 +19,5 @@ dependencies { implementation(libs.orbit.viewmodel) implementation(libs.androidx.material.android) implementation(libs.androidx.annotation) + implementation(libs.coil.compose) } diff --git a/feature/home/src/main/AndroidManifest.xml b/feature/home/src/main/AndroidManifest.xml index 8bdb7e14..1c6dcf4e 100644 --- a/feature/home/src/main/AndroidManifest.xml +++ b/feature/home/src/main/AndroidManifest.xml @@ -1,4 +1,4 @@ - + diff --git a/feature/home/src/main/java/com/yapp/home/HomeContract.kt b/feature/home/src/main/java/com/yapp/home/HomeContract.kt index dee0ecef..e4d15d19 100644 --- a/feature/home/src/main/java/com/yapp/home/HomeContract.kt +++ b/feature/home/src/main/java/com/yapp/home/HomeContract.kt @@ -18,6 +18,7 @@ sealed class HomeContract { val isDeleteDialogVisible: Boolean = false, val isNoActivatedAlarmDialogVisible: Boolean = false, val isNoDailyFortuneDialogVisible: Boolean = false, + val isUpdateNoticeVisible: Boolean = false, val hasNewFortune: Boolean = false, val isToolTipVisible: Boolean = false, val pendingAlarmToggle: Pair? = null, @@ -58,6 +59,8 @@ sealed class HomeContract { data object ShowNoDailyFortuneDialog : Action() data object HideNoDailyFortuneDialog : Action() data object HideToolTip : Action() + data object OnClickDontShowAgain : Action() + data object HideUpdateNotice : Action() data object RollbackPendingAlarmToggle : Action() data object ConfirmDeletion : Action() data class DeleteSingleAlarm(val alarmId: Long) : Action() diff --git a/feature/home/src/main/java/com/yapp/home/HomeScreen.kt b/feature/home/src/main/java/com/yapp/home/HomeScreen.kt index 8c44e05a..10d74e52 100644 --- a/feature/home/src/main/java/com/yapp/home/HomeScreen.kt +++ b/feature/home/src/main/java/com/yapp/home/HomeScreen.kt @@ -1,5 +1,6 @@ package com.yapp.home +import android.os.Build import androidx.compose.animation.core.animateDpAsState import androidx.compose.animation.core.tween import androidx.compose.foundation.Canvas @@ -74,6 +75,7 @@ import com.yapp.domain.model.Alarm import com.yapp.home.alarm.component.AlarmListItem import com.yapp.home.alarm.component.AlarmListItemMenu import com.yapp.home.component.bottomsheet.AlarmListBottomSheet +import com.yapp.home.component.bottomsheet.UpdateNoticeBottomSheet import com.yapp.ui.component.dialog.OrbitDialog import com.yapp.ui.component.lottie.LottieAnimation import com.yapp.ui.component.snackbar.showCustomSnackBar @@ -129,8 +131,8 @@ fun HomeRoute( } HomeScreen( - stateProvider = { state }, - eventDispatcher = viewModel::processAction, + state = state, + processAction = viewModel::processAction, ) } @@ -178,23 +180,21 @@ private suspend fun handleSideEffect( @Composable fun HomeScreen( - stateProvider: () -> HomeContract.State, - eventDispatcher: (HomeContract.Action) -> Unit, + state: HomeContract.State, + processAction: (HomeContract.Action) -> Unit, ) { - val state = stateProvider() - if (state.initialLoading) { HomeLoadingScreen() } else if (state.alarms.isEmpty()) { HomeAlarmEmptyScreen( onSettingClick = { - eventDispatcher(HomeContract.Action.NavigateToSetting) + processAction(HomeContract.Action.NavigateToSetting) }, onMailClick = { - eventDispatcher(HomeContract.Action.ShowDailyFortune) + processAction(HomeContract.Action.ShowDailyFortune) }, onAddClick = { - eventDispatcher(HomeContract.Action.NavigateToAlarmCreation) + processAction(HomeContract.Action.NavigateToAlarmCreation) }, hasNewFortune = state.hasNewFortune, isTooltipVisible = state.isToolTipVisible, @@ -202,7 +202,7 @@ fun HomeScreen( } else { HomeContent( state = state, - eventDispatcher = eventDispatcher, + processAction = processAction, ) } @@ -213,10 +213,10 @@ fun HomeScreen( confirmText = stringResource(id = R.string.alarm_delete_dialog_btn_delete), cancelText = stringResource(id = R.string.alarm_delete_dialog_btn_cancel), onConfirm = { - eventDispatcher(HomeContract.Action.ConfirmDeletion) + processAction(HomeContract.Action.ConfirmDeletion) }, onCancel = { - eventDispatcher(HomeContract.Action.HideDeleteDialog) + processAction(HomeContract.Action.HideDeleteDialog) }, ) } @@ -228,10 +228,10 @@ fun HomeScreen( confirmText = stringResource(id = R.string.no_active_alarm_dialog_btn_confirm), cancelText = stringResource(id = R.string.no_active_alarm_dialog_btn_cancel), onConfirm = { - eventDispatcher(HomeContract.Action.HideNoActivatedAlarmDialog) + processAction(HomeContract.Action.HideNoActivatedAlarmDialog) }, onCancel = { - eventDispatcher(HomeContract.Action.RollbackPendingAlarmToggle) + processAction(HomeContract.Action.RollbackPendingAlarmToggle) }, ) } @@ -242,7 +242,18 @@ fun HomeScreen( message = stringResource(id = R.string.no_daily_fortune_dialog_message), confirmText = stringResource(id = R.string.no_daily_fortune_dialog_btn_confirm), onConfirm = { - eventDispatcher(HomeContract.Action.HideNoDailyFortuneDialog) + processAction(HomeContract.Action.HideNoDailyFortuneDialog) + }, + ) + } + + if (state.isUpdateNoticeVisible) { + UpdateNoticeBottomSheet( + onDontShowAgain = { + processAction(HomeContract.Action.OnClickDontShowAgain) + }, + onClose = { + processAction(HomeContract.Action.HideUpdateNotice) }, ) } @@ -279,7 +290,7 @@ private fun HomeLoadingScreen() { @Composable private fun HomeContent( state: HomeContract.State, - eventDispatcher: (HomeContract.Action) -> Unit, + processAction: (HomeContract.Action) -> Unit, ) { val screenHeight = LocalConfiguration.current.screenHeightDp.dp val statusBarHeight = WindowInsets.statusBars.asPaddingValues().calculateTopPadding() @@ -291,7 +302,7 @@ private fun HomeContent( LaunchedEffect(state.lastAddedAlarmIndex) { state.lastAddedAlarmIndex?.let { index -> listState.scrollToItem(index) - eventDispatcher(HomeContract.Action.ResetLastAddedAlarmIndex) + processAction(HomeContract.Action.ResetLastAddedAlarmIndex) } } @@ -301,7 +312,7 @@ private fun HomeContent( interactionSource = remember { MutableInteractionSource() }, indication = null, ) { - eventDispatcher(HomeContract.Action.HideToolTip) + processAction(HomeContract.Action.HideToolTip) }, ) { if (state.activeItemMenu != null) { @@ -312,7 +323,7 @@ private fun HomeContent( interactionSource = remember { MutableInteractionSource() }, indication = null, ) { - eventDispatcher(HomeContract.Action.HideItemMenu) + processAction(HomeContract.Action.HideItemMenu) } .zIndex(1f), ) @@ -338,47 +349,50 @@ private fun HomeContent( halfExpandedHeight = sheetHalfExpandHeight, listState = listState, onClickAlarm = { alarmId -> - eventDispatcher(HomeContract.Action.EditAlarm(alarmId)) + processAction(HomeContract.Action.EditAlarm(alarmId)) }, onLongPressAlarm = { alarmId, x, y -> - eventDispatcher(HomeContract.Action.ShowItemMenu(alarmId, x, y)) + processAction(HomeContract.Action.ShowItemMenu(alarmId, x, y)) }, onClickAdd = { - eventDispatcher(HomeContract.Action.NavigateToAlarmCreation) + processAction(HomeContract.Action.NavigateToAlarmCreation) }, onClickMore = { if (state.dropdownMenuExpanded || state.sortDropDownMenuExpanded) { - eventDispatcher(HomeContract.Action.HideDropDownMenu) + processAction(HomeContract.Action.HideDropDownMenu) } else { - eventDispatcher(HomeContract.Action.ShowDropDownMenu) + processAction(HomeContract.Action.ShowDropDownMenu) } }, onClickCheckAll = { - eventDispatcher(HomeContract.Action.ToggleAllAlarmSelection) + processAction(HomeContract.Action.ToggleAllAlarmSelection) }, onClickClose = { - eventDispatcher(HomeContract.Action.ToggleMultiSelectionMode) + processAction(HomeContract.Action.ToggleMultiSelectionMode) }, onClickEdit = { - eventDispatcher(HomeContract.Action.ToggleMultiSelectionMode) + processAction(HomeContract.Action.ToggleMultiSelectionMode) }, onClickSort = { - eventDispatcher(HomeContract.Action.ShowSortDropDownMenu) + processAction(HomeContract.Action.ShowSortDropDownMenu) }, onSetSortOrder = { sortOrder -> - eventDispatcher(HomeContract.Action.SetSortOrder(sortOrder)) + processAction(HomeContract.Action.SetSortOrder(sortOrder)) }, onDismissRequest = { - eventDispatcher(HomeContract.Action.HideDropDownMenu) + processAction(HomeContract.Action.HideDropDownMenu) }, onToggleSelect = { alarmId -> - eventDispatcher(HomeContract.Action.ToggleAlarmSelection(alarmId)) + processAction(HomeContract.Action.ToggleAlarmSelection(alarmId)) }, onToggleActive = { alarmId -> - eventDispatcher(HomeContract.Action.ToggleAlarmActivation(alarmId)) + processAction(HomeContract.Action.ToggleAlarmActivation(alarmId)) }, onSwipe = { alarmId -> - eventDispatcher(HomeContract.Action.SwipeToDeleteAlarm(alarmId)) + processAction(HomeContract.Action.SwipeToDeleteAlarm(alarmId)) + }, + onExpanded = { + processAction(HomeContract.Action.HideToolTip) }, ) { Box( @@ -397,7 +411,15 @@ private fun HomeContent( .fillMaxWidth() .layout { measurable, constraints -> val placeable = measurable.measure(constraints) - sheetHalfExpandHeight = screenHeight - placeable.height.toDp() - statusBarHeight - navBarHeight + val contentHeight = placeable.height.toDp() + + val offset = if (Build.VERSION.SDK_INT < 35) { + 0.dp + } else { + statusBarHeight + navBarHeight + } + sheetHalfExpandHeight = screenHeight - contentHeight - offset + layout(placeable.width, placeable.height) { placeable.placeRelative(0, 0) } @@ -420,8 +442,8 @@ private fun HomeContent( } HomeTopBar( - onSettingClick = { eventDispatcher(HomeContract.Action.NavigateToSetting) }, - onMailClick = { eventDispatcher(HomeContract.Action.ShowDailyFortune) }, + onSettingClick = { processAction(HomeContract.Action.NavigateToSetting) }, + onMailClick = { processAction(HomeContract.Action.ShowDailyFortune) }, hasNewFortune = state.hasNewFortune, isShowTooltip = state.isToolTipVisible, ) @@ -438,7 +460,7 @@ private fun HomeContent( .padding(bottom = 26.dp), selectedAlarmCount = state.selectedAlarmIds.size, onClick = { - eventDispatcher(HomeContract.Action.ShowDeleteDialog) + processAction(HomeContract.Action.ShowDeleteDialog) }, ) } @@ -451,7 +473,7 @@ private fun HomeContent( activeItemMenuPosition = state.activeItemMenuPosition, selectedAlarmIds = state.selectedAlarmIds, onDelete = { alarmId -> - eventDispatcher(HomeContract.Action.DeleteSingleAlarm(alarmId)) + processAction(HomeContract.Action.DeleteSingleAlarm(alarmId)) }, ) } @@ -938,18 +960,16 @@ private fun AlarmWithMenu( fun HomeScreenPreview() { OrbitTheme { HomeScreen( - stateProvider = { - HomeContract.State() - .copy( - initialLoading = false, - alarms = listOf( - Alarm(), - ), - activeItemMenu = 0L, - activeItemMenuPosition = Pair(0f, 0f), - ) - }, - eventDispatcher = {}, + state = HomeContract.State() + .copy( + initialLoading = false, + alarms = listOf( + Alarm(), + ), + activeItemMenu = 0L, + activeItemMenuPosition = Pair(0f, 0f), + ), + processAction = {}, ) } } diff --git a/feature/home/src/main/java/com/yapp/home/HomeViewModel.kt b/feature/home/src/main/java/com/yapp/home/HomeViewModel.kt index ae347cea..57b74428 100644 --- a/feature/home/src/main/java/com/yapp/home/HomeViewModel.kt +++ b/feature/home/src/main/java/com/yapp/home/HomeViewModel.kt @@ -1,5 +1,8 @@ package com.yapp.home +import android.content.Context +import android.net.ConnectivityManager +import android.net.NetworkCapabilities import android.util.Log import androidx.lifecycle.ViewModel import com.yapp.common.util.ResourceProvider @@ -9,6 +12,7 @@ import com.yapp.domain.repository.UserInfoRepository import com.yapp.domain.usecase.AlarmUseCase import com.yapp.home.util.AlarmDateTimeFormatter import dagger.hilt.android.lifecycle.HiltViewModel +import dagger.hilt.android.qualifiers.ApplicationContext import feature.home.R import kotlinx.coroutines.flow.combine import kotlinx.coroutines.flow.first @@ -21,8 +25,8 @@ import org.orbitmvi.orbit.syntax.simple.reduce import org.orbitmvi.orbit.syntax.simple.repeatOnSubscription import org.orbitmvi.orbit.viewmodel.container import java.time.LocalDate -import java.time.format.DateTimeFormatter import javax.inject.Inject +import javax.inject.Named @HiltViewModel class HomeViewModel @Inject constructor( @@ -31,6 +35,8 @@ class HomeViewModel @Inject constructor( private val alarmDateTimeFormatter: AlarmDateTimeFormatter, private val fortuneRepository: FortuneRepository, private val userInfoRepository: UserInfoRepository, + @Named("appVersion") private val appVersion: String, + @ApplicationContext private val context: Context, ) : ViewModel(), ContainerHost { override val container: Container = container( @@ -41,6 +47,7 @@ class HomeViewModel @Inject constructor( loadAllAlarms() loadDailyFortuneState() loadUserName() + loadUpdateNoticeVisibility() } } } @@ -63,6 +70,8 @@ class HomeViewModel @Inject constructor( HomeContract.Action.ShowNoDailyFortuneDialog -> showNoDailyFortuneDialog() HomeContract.Action.HideNoDailyFortuneDialog -> hideNoDailyFortuneDialog() HomeContract.Action.HideToolTip -> hideToolTip() + HomeContract.Action.HideUpdateNotice -> hideUpdateNotice() + HomeContract.Action.OnClickDontShowAgain -> setUpdateNoticeDontShowVersion() HomeContract.Action.RollbackPendingAlarmToggle -> rollbackAlarmActivation() HomeContract.Action.ConfirmDeletion -> confirmDeletion() is HomeContract.Action.DeleteSingleAlarm -> deleteSingleAlarm(action.alarmId) @@ -332,29 +341,29 @@ class HomeViewModel @Inject constructor( } private fun loadDailyFortune() = intent { - val fortuneDate = fortuneRepository.fortuneDateFlow.firstOrNull() - val todayDate = LocalDate.now().format(DateTimeFormatter.ISO_DATE) + val fortuneDate = fortuneRepository.fortuneDateEpochFlow.firstOrNull() + val todayDate = LocalDate.now().toEpochDay() if (fortuneDate != todayDate) { processAction(HomeContract.Action.ShowNoDailyFortuneDialog) } else { - fortuneRepository.markFortuneAsChecked() + fortuneRepository.markFortuneTooltipShown() postSideEffect(HomeContract.SideEffect.NavigateToFortune) } } private fun loadDailyFortuneState() = intent { - val todayDate = LocalDate.now().format(DateTimeFormatter.ISO_DATE) + val todayDate = LocalDate.now().toEpochDay() combine( - fortuneRepository.fortuneDateFlow, + fortuneRepository.fortuneDateEpochFlow, fortuneRepository.fortuneScoreFlow, - fortuneRepository.hasNewFortuneFlow, - ) { fortuneDate, fortuneScore, hasNewFortune -> + fortuneRepository.shouldShowFortuneToolTipFlow, + ) { fortuneDate, fortuneScore, shouldShowTooltip -> val isTodayFortuneAvailable = fortuneDate == todayDate val finalFortuneScore = if (isTodayFortuneAvailable) fortuneScore ?: -1 else -1 - Pair(finalFortuneScore, hasNewFortune) + Pair(finalFortuneScore, shouldShowTooltip) }.collect { (finalFortuneScore, hasNewFortune) -> reduce { state.copy( @@ -366,6 +375,39 @@ class HomeViewModel @Inject constructor( } } + private fun loadUpdateNoticeVisibility() = intent { + if (!isOnlineNow()) { + reduce { state.copy(isUpdateNoticeVisible = false) } + return@intent + } + + val dontShowVersion = + userInfoRepository.updateNoticeDontShowVersionFlow.firstOrNull() + val lastShownDate = + userInfoRepository.updateNoticeLastShownDateEpochFlow.firstOrNull() + + val today = LocalDate.now().toEpochDay() + + val shouldShow = when { + dontShowVersion != null && dontShowVersion == appVersion -> false + lastShownDate != null && lastShownDate == today -> false + else -> true + } + + if (shouldShow) userInfoRepository.markUpdateNoticeShownToday() + + reduce { state.copy(isUpdateNoticeVisible = shouldShow) } + } + + private fun setUpdateNoticeDontShowVersion() = intent { + userInfoRepository.markUpdateNoticeDontShow(appVersion) + reduce { state.copy(isUpdateNoticeVisible = false) } + } + + private fun hideUpdateNotice() = intent { + reduce { state.copy(isUpdateNoticeVisible = false) } + } + private fun loadUserName() = intent { userInfoRepository.userNameFlow.first { userName -> reduce { state.copy(name = userName ?: "") } @@ -411,4 +453,13 @@ class HomeViewModel @Inject constructor( reduce { state.copy(sortOrder = sortOrder) } hideDropDownMenu() } + + private fun isOnlineNow(): Boolean { + val cm = context.getSystemService(Context.CONNECTIVITY_SERVICE) as ConnectivityManager + val network = cm.activeNetwork ?: return false + val caps = cm.getNetworkCapabilities(network) ?: return false + + return caps.hasCapability(NetworkCapabilities.NET_CAPABILITY_INTERNET) && + caps.hasCapability(NetworkCapabilities.NET_CAPABILITY_VALIDATED) + } } diff --git a/feature/home/src/main/java/com/yapp/home/alarm/addedit/AlarmAddEditScreen.kt b/feature/home/src/main/java/com/yapp/home/alarm/addedit/AlarmAddEditScreen.kt index ec6dcefe..9d726ae3 100644 --- a/feature/home/src/main/java/com/yapp/home/alarm/addedit/AlarmAddEditScreen.kt +++ b/feature/home/src/main/java/com/yapp/home/alarm/addedit/AlarmAddEditScreen.kt @@ -61,6 +61,7 @@ import com.yapp.home.alarm.component.bottomsheet.AlarmMissionBottomSheet import com.yapp.home.alarm.component.bottomsheet.AlarmSnoozeBottomSheet import com.yapp.home.alarm.component.bottomsheet.AlarmSoundBottomSheet import com.yapp.home.alarm.getLabelStringRes +import com.yapp.ui.component.bottomsheet.OrbitBottomSheetLayout import com.yapp.ui.component.bottomsheet.OrbitBottomSheetState import com.yapp.ui.component.bottomsheet.rememberOrbitBottomSheetState import com.yapp.ui.component.button.OrbitButton @@ -282,50 +283,52 @@ fun AlarmAddEditContent( } } - Column( - modifier = Modifier.fillMaxSize(), - horizontalAlignment = Alignment.CenterHorizontally, - ) { - AlarmAddEditTopBar( - mode = state.mode, - title = state.timeState.alarmMessage, - onBack = { processAction(AlarmAddEditContract.Action.CheckUnsavedChangesBeforeExit) }, - onDelete = { processAction(AlarmAddEditContract.Action.ShowDeleteDialog) }, - ) - Box( - modifier = Modifier.weight(1f), - contentAlignment = Alignment.Center, + OrbitBottomSheetLayout(sheetState = bottomSheetState) { + Column( + modifier = Modifier.fillMaxSize(), + horizontalAlignment = Alignment.CenterHorizontally, ) { - OrbitPicker( - initialTime = state.timeState.initialTime, - ) { newTime -> - processAction(AlarmAddEditContract.Action.SetAlarmTime(newTime)) + AlarmAddEditTopBar( + mode = state.mode, + title = state.timeState.alarmMessage, + onBack = { processAction(AlarmAddEditContract.Action.CheckUnsavedChangesBeforeExit) }, + onDelete = { processAction(AlarmAddEditContract.Action.ShowDeleteDialog) }, + ) + Box( + modifier = Modifier.weight(1f), + contentAlignment = Alignment.Center, + ) { + OrbitPicker( + initialTime = state.timeState.initialTime, + ) { newTime -> + processAction(AlarmAddEditContract.Action.SetAlarmTime(newTime)) + } } + AlarmAddEditSelectDaysSection( + modifier = Modifier.padding(horizontal = 20.dp), + daysSelectionState = state.daySelectionState, + holidayState = state.holidayState, + processAction = processAction, + ) + Spacer(modifier = Modifier.height(12.dp)) + AlarmAddEditSettingsSection( + modifier = Modifier.padding(horizontal = 20.dp), + state = state, + processAction = processAction, + ) + Spacer(modifier = Modifier.height(24.dp)) + OrbitButton( + label = stringResource(R.string.alarm_add_edit_save), + onClick = { processAction(AlarmAddEditContract.Action.SaveAlarm) }, + enabled = true, + modifier = Modifier + .padding( + start = 20.dp, + end = 20.dp, + bottom = 12.dp, + ), + ) } - AlarmAddEditSelectDaysSection( - modifier = Modifier.padding(horizontal = 20.dp), - daysSelectionState = state.daySelectionState, - holidayState = state.holidayState, - processAction = processAction, - ) - Spacer(modifier = Modifier.height(12.dp)) - AlarmAddEditSettingsSection( - modifier = Modifier.padding(horizontal = 20.dp), - state = state, - processAction = processAction, - ) - Spacer(modifier = Modifier.height(24.dp)) - OrbitButton( - label = stringResource(R.string.alarm_add_edit_save), - onClick = { processAction(AlarmAddEditContract.Action.SaveAlarm) }, - enabled = true, - modifier = Modifier - .padding( - start = 20.dp, - end = 20.dp, - bottom = 12.dp, - ), - ) } if (state.isDeleteDialogVisible) { diff --git a/feature/home/src/main/java/com/yapp/home/alarm/component/bottomsheet/AlarmMissionBottomSheet.kt b/feature/home/src/main/java/com/yapp/home/alarm/component/bottomsheet/AlarmMissionBottomSheet.kt index 0c84f840..99656263 100644 --- a/feature/home/src/main/java/com/yapp/home/alarm/component/bottomsheet/AlarmMissionBottomSheet.kt +++ b/feature/home/src/main/java/com/yapp/home/alarm/component/bottomsheet/AlarmMissionBottomSheet.kt @@ -21,10 +21,13 @@ import androidx.compose.material3.ExperimentalMaterial3Api import androidx.compose.material3.Icon import androidx.compose.material3.Text import androidx.compose.runtime.Composable +import androidx.compose.runtime.MutableState import androidx.compose.runtime.getValue import androidx.compose.runtime.mutableIntStateOf import androidx.compose.runtime.mutableStateOf -import androidx.compose.runtime.remember +import androidx.compose.runtime.saveable.Saver +import androidx.compose.runtime.saveable.listSaver +import androidx.compose.runtime.saveable.rememberSaveable import androidx.compose.runtime.setValue import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier @@ -57,6 +60,12 @@ private fun MissionType.displayData(): Pair = when (this) { else -> throw IllegalStateException("Invalid mission type") } +val StepStackSaver: Saver>, out Any> = + listSaver( + save = { state -> state.value.map { it.name } }, + restore = { restored -> mutableStateOf(restored.map { AlarmMissionSelectBottomSheetType.valueOf(it) }) }, + ) + @Composable internal fun AlarmMissionBottomSheet( missionState: AlarmAddEditContract.AlarmMissionState, @@ -64,9 +73,15 @@ internal fun AlarmMissionBottomSheet( onSaveMission: (MissionType, Int) -> Unit, onPreviewMission: (MissionType, Int) -> Unit, ) { - var stepStack by remember { mutableStateOf(listOf(AlarmMissionSelectBottomSheetType.MISSION_SETTING)) } - var selectedMissionType by remember { mutableStateOf(missionState.missionType) } - var selectedMissionCount by remember { mutableIntStateOf(missionState.missionCount) } + val initialMissionType = missionState.missionType + val initialMissionCount = missionState.missionCount + + var stepStack by rememberSaveable(saver = StepStackSaver) { + mutableStateOf(listOf(AlarmMissionSelectBottomSheetType.MISSION_SETTING)) + } + + var currentSelectedMissionType by rememberSaveable { mutableStateOf(initialMissionType) } + var currentSelectedMissionCount by rememberSaveable { mutableIntStateOf(initialMissionCount) } fun push(step: AlarmMissionSelectBottomSheetType) { stepStack = stepStack + step @@ -82,22 +97,22 @@ internal fun AlarmMissionBottomSheet( when (currentStep) { AlarmMissionSelectBottomSheetType.MISSION_SETTING -> { - if (selectedMissionType == MissionType.NONE) { + if (currentSelectedMissionType == MissionType.NONE) { MissionAddContent { push(AlarmMissionSelectBottomSheetType.MISSION_SELECT) } } else { MissionSettingContent( - missionType = selectedMissionType, - missionCount = selectedMissionCount, + missionType = currentSelectedMissionType, + missionCount = currentSelectedMissionCount, onDetail = { push(AlarmMissionSelectBottomSheetType.MISSION_DETAIL) }, onDelete = { - selectedMissionType = MissionType.NONE - onSaveMission(selectedMissionType, selectedMissionCount) + currentSelectedMissionType = MissionType.NONE + onSaveMission(currentSelectedMissionType, currentSelectedMissionCount) }, onChange = { push(AlarmMissionSelectBottomSheetType.MISSION_SELECT) }, onDone = { - onSaveMission(selectedMissionType, selectedMissionCount) + onSaveMission(currentSelectedMissionType, currentSelectedMissionCount) onDismiss() }, ) @@ -107,11 +122,10 @@ internal fun AlarmMissionBottomSheet( AlarmMissionSelectBottomSheetType.MISSION_SELECT -> { MissionSelectContent( onBack = { pop() }, - onClose = { - onDismiss() - }, - onSelect = { mission -> - selectedMissionType = mission + onClose = { onDismiss() }, + initialMission = currentSelectedMissionType, + onSelect = { selected -> + currentSelectedMissionType = selected push(AlarmMissionSelectBottomSheetType.MISSION_DETAIL) }, ) @@ -119,19 +133,17 @@ internal fun AlarmMissionBottomSheet( AlarmMissionSelectBottomSheetType.MISSION_DETAIL -> { MissionDetailContent( - missionType = selectedMissionType, - selectedMissionCount = selectedMissionCount, - onCountChange = { selectedMissionCount = it }, + missionType = currentSelectedMissionType, + selectedMissionCount = currentSelectedMissionCount, + onCountChange = { currentSelectedMissionCount = it }, onBack = { pop() }, - onClose = { - onDismiss() - }, + onClose = { onDismiss() }, onSave = { - onSaveMission(selectedMissionType, selectedMissionCount) + onSaveMission(currentSelectedMissionType, currentSelectedMissionCount) onDismiss() }, onPreview = { - onPreviewMission(selectedMissionType, selectedMissionCount) + onPreviewMission(currentSelectedMissionType, currentSelectedMissionCount) }, ) } @@ -379,6 +391,7 @@ private fun MissionCountChip( private fun MissionSelectContent( onBack: () -> Unit, onClose: () -> Unit, + initialMission: MissionType, onSelect: (MissionType) -> Unit, ) { Column( @@ -390,7 +403,7 @@ private fun MissionSelectContent( Spacer(modifier = Modifier.height(14.dp)) MissionSelectTopAppBar( - title = stringResource(id = feature.home.R.string.mission_bottom_sheet_title), + title = stringResource(id = feature.home.R.string.mission_select_content_title), onBack = onBack, onClose = onClose, ) @@ -400,12 +413,14 @@ private fun MissionSelectContent( ) { MissionTypeItem( missionType = MissionType.SHAKE, + selected = initialMission == MissionType.SHAKE, onClick = { onSelect(MissionType.SHAKE) }, ) MissionTypeItem( missionType = MissionType.TAP, + selected = initialMission == MissionType.TAP, onClick = { onSelect(MissionType.TAP) }, @@ -417,6 +432,7 @@ private fun MissionSelectContent( @Composable private fun MissionTypeItem( missionType: MissionType, + selected: Boolean, onClick: () -> Unit, ) { val (iconRes, titleRes) = missionType.displayData() @@ -425,6 +441,7 @@ private fun MissionTypeItem( Row( modifier = Modifier .fillMaxWidth() + .clip(RoundedCornerShape(12.dp)) .clickable( onClick = onClick, ) @@ -447,6 +464,28 @@ private fun MissionTypeItem( style = OrbitTheme.typography.headline2SemiBold, color = OrbitTheme.colors.white, ) + + if (selected) { + Spacer(modifier = Modifier.weight(1f)) + + Row( + horizontalArrangement = Arrangement.spacedBy(2.dp), + verticalAlignment = Alignment.CenterVertically, + ) { + Icon( + modifier = Modifier.size(16.dp), + painter = painterResource(id = R.drawable.ic_check), + tint = OrbitTheme.colors.white.copy(alpha = 0.5f), + contentDescription = null, + ) + + Text( + text = stringResource(id = feature.home.R.string.mission_select_content_selected), + style = OrbitTheme.typography.body2Medium, + color = OrbitTheme.colors.white.copy(alpha = 0.4f), + ) + } + } } } diff --git a/feature/home/src/main/java/com/yapp/home/component/bottomsheet/AlarmListBottomSheet.kt b/feature/home/src/main/java/com/yapp/home/component/bottomsheet/AlarmListBottomSheet.kt index 9aaec0b3..e8ee3c58 100644 --- a/feature/home/src/main/java/com/yapp/home/component/bottomsheet/AlarmListBottomSheet.kt +++ b/feature/home/src/main/java/com/yapp/home/component/bottomsheet/AlarmListBottomSheet.kt @@ -36,6 +36,7 @@ import androidx.compose.runtime.mutableStateOf import androidx.compose.runtime.remember import androidx.compose.runtime.rememberCoroutineScope import androidx.compose.runtime.setValue +import androidx.compose.runtime.snapshotFlow import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.draw.clip @@ -56,6 +57,8 @@ import com.yapp.home.component.AlarmListDropDownMenu import com.yapp.home.component.AlarmSortDropDownMenu import com.yapp.ui.component.checkbox.OrbitCheckBox import feature.home.R +import kotlinx.coroutines.flow.collectLatest +import kotlinx.coroutines.flow.distinctUntilChanged import kotlinx.coroutines.launch enum class BottomSheetExpandState { @@ -87,23 +90,29 @@ internal fun AlarmListBottomSheet( onToggleSelect: (Long) -> Unit, onToggleActive: (Long) -> Unit, onSwipe: (Long) -> Unit, + onExpanded: () -> Unit, content: @Composable () -> Unit, ) { var expandedType by remember { mutableStateOf(BottomSheetExpandState.HALF_EXPANDED) } - val sheetState = rememberStandardBottomSheetState( - confirmValueChange = { - expandedType = when (it) { - SheetValue.Expanded -> BottomSheetExpandState.EXPANDED - else -> BottomSheetExpandState.HALF_EXPANDED - } - true - }, - ) - + val sheetState = rememberStandardBottomSheetState() val scaffoldState = rememberBottomSheetScaffoldState(bottomSheetState = sheetState) val coroutineScope = rememberCoroutineScope() + LaunchedEffect(Unit) { + snapshotFlow { sheetState.currentValue } + .distinctUntilChanged() + .collectLatest { value -> + expandedType = when (value) { + SheetValue.Expanded -> { + onExpanded() + BottomSheetExpandState.EXPANDED + } + SheetValue.PartiallyExpanded, SheetValue.Hidden -> BottomSheetExpandState.HALF_EXPANDED + } + } + } + val nestedScrollConnection = remember { object : androidx.compose.ui.input.nestedscroll.NestedScrollConnection { override fun onPreScroll( @@ -118,13 +127,6 @@ internal fun AlarmListBottomSheet( } } - LaunchedEffect(sheetState.currentValue) { - expandedType = when (sheetState.currentValue) { - SheetValue.Expanded -> BottomSheetExpandState.EXPANDED - else -> BottomSheetExpandState.HALF_EXPANDED - } - } - BottomSheetScaffold( scaffoldState = scaffoldState, sheetContent = { @@ -324,23 +326,19 @@ private fun AlarmListTopBar( onClick = onClickMore, ) - if (menuExpanded) { - AlarmListDropDownMenu( - expanded = menuExpanded, - onDismissRequest = onDismissRequest, - onClickEdit = onClickEdit, - onClickSort = onClickSort, - ) - } + AlarmListDropDownMenu( + expanded = menuExpanded, + onDismissRequest = onDismissRequest, + onClickEdit = onClickEdit, + onClickSort = onClickSort, + ) - if (sortDropDownMenuExpanded) { - AlarmSortDropDownMenu( - expanded = sortDropDownMenuExpanded, - sortOrder = sortOrder, - onDismissRequest = onDismissRequest, - onSetSortOrder = onSetSortOrder, - ) - } + AlarmSortDropDownMenu( + expanded = sortDropDownMenuExpanded, + sortOrder = sortOrder, + onDismissRequest = onDismissRequest, + onSetSortOrder = onSetSortOrder, + ) } } } diff --git a/feature/home/src/main/java/com/yapp/home/component/bottomsheet/UpdateNoticeBottomSheet.kt b/feature/home/src/main/java/com/yapp/home/component/bottomsheet/UpdateNoticeBottomSheet.kt new file mode 100644 index 00000000..fd38d5c2 --- /dev/null +++ b/feature/home/src/main/java/com/yapp/home/component/bottomsheet/UpdateNoticeBottomSheet.kt @@ -0,0 +1,150 @@ +package com.yapp.home.component.bottomsheet + +import android.content.pm.PackageManager +import android.os.Build +import androidx.compose.foundation.background +import androidx.compose.foundation.clickable +import androidx.compose.foundation.interaction.MutableInteractionSource +import androidx.compose.foundation.layout.Arrangement +import androidx.compose.foundation.layout.Box +import androidx.compose.foundation.layout.Column +import androidx.compose.foundation.layout.Row +import androidx.compose.foundation.layout.aspectRatio +import androidx.compose.foundation.layout.fillMaxSize +import androidx.compose.foundation.layout.fillMaxWidth +import androidx.compose.foundation.layout.padding +import androidx.compose.foundation.shape.RoundedCornerShape +import androidx.compose.material3.Text +import androidx.compose.runtime.Composable +import androidx.compose.runtime.remember +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.platform.LocalContext +import androidx.compose.ui.platform.LocalInspectionMode +import androidx.compose.ui.res.stringResource +import androidx.compose.ui.tooling.preview.Preview +import androidx.compose.ui.unit.dp +import coil.compose.AsyncImage +import com.yapp.designsystem.theme.OrbitTheme +import feature.home.R + +private fun resolveVersionName(ctx: android.content.Context): String { + return runCatching { + val pm = ctx.packageManager + val packageName = ctx.packageName + val info = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) { + pm.getPackageInfo(packageName, PackageManager.PackageInfoFlags.of(0)) + } else { + @Suppress("DEPRECATION") + pm.getPackageInfo(packageName, 0) + } + info.versionName ?: "" + }.getOrDefault("") +} + +private fun bannerUrl(versionName: String): String = + "https://www.orbitalarm.net/images/aos/$versionName/update-banner.png" + +@Composable +internal fun UpdateNoticeBottomSheet( + onDontShowAgain: () -> Unit, + onClose: () -> Unit, +) { + val context = LocalContext.current + val isPreview = LocalInspectionMode.current + + val versionName = remember(isPreview) { + if (isPreview) "preview" else resolveVersionName(context) + } + val imageUrl = remember(versionName) { bannerUrl(versionName.ifEmpty { "unknown" }) } + + Box( + modifier = Modifier + .fillMaxSize() + .background(Color(0xFF17191F).copy(alpha = 0.85f)) + .clickable(onClick = onClose), + contentAlignment = Alignment.BottomCenter, + ) { + Column( + modifier = Modifier + .fillMaxWidth() + .background( + color = OrbitTheme.colors.gray_900, + shape = RoundedCornerShape(topStart = 30.dp, topEnd = 30.dp), + ) + .clip(RoundedCornerShape(topStart = 30.dp, topEnd = 30.dp)) + .clickable( + indication = null, + interactionSource = remember { MutableInteractionSource() }, + ) { }, + ) { + if (isPreview) { + // 프리뷰용 박스 + Box( + modifier = Modifier + .fillMaxWidth() + .aspectRatio(1f) + .background(color = OrbitTheme.colors.white), + ) + } else { + AsyncImage( + model = imageUrl, + contentDescription = null, + contentScale = ContentScale.Crop, + modifier = Modifier + .fillMaxWidth() + .aspectRatio(1f), + ) + } + + Row( + modifier = Modifier + .fillMaxWidth() + .padding(top = 8.dp, bottom = 20.dp, start = 20.dp, end = 20.dp), + horizontalArrangement = Arrangement.spacedBy(10.dp), + verticalAlignment = Alignment.CenterVertically, + ) { + Box( + modifier = Modifier + .weight(1f) + .clickable(onClick = onDontShowAgain) + .padding(vertical = 14.dp), + contentAlignment = Alignment.Center, + ) { + Text( + text = stringResource(id = R.string.update_notice_bottom_sheet_dont_show_again), + style = OrbitTheme.typography.body1SemiBold, + color = OrbitTheme.colors.white, + ) + } + Box( + modifier = Modifier + .weight(1f) + .clickable(onClick = onClose) + .padding(vertical = 14.dp), + contentAlignment = Alignment.Center, + ) { + Text( + text = stringResource(id = R.string.update_notice_bottom_sheet_close), + style = OrbitTheme.typography.body1SemiBold, + color = OrbitTheme.colors.white, + ) + } + } + } + } +} + +@Preview +@Composable +private fun UpdateNoticeBottomSheetPreview() { + OrbitTheme { + UpdateNoticeBottomSheet( + onDontShowAgain = {}, + onClose = {}, + ) + } +} diff --git a/feature/home/src/main/res/values/strings.xml b/feature/home/src/main/res/values/strings.xml index cf91a124..b2266b16 100644 --- a/feature/home/src/main/res/values/strings.xml +++ b/feature/home/src/main/res/values/strings.xml @@ -48,6 +48,9 @@ 미션 변경 완료 + 미션 선택 + 선택됨 + %d회 횟수 @@ -121,4 +124,7 @@ %1$d시간 %2$d분 후에 울려요 %d분 후에 울려요 곧 울려요 + + 다시 보지 않기 + 닫기 diff --git a/feature/mission/src/main/java/com/yapp/mission/MissionContract.kt b/feature/mission/src/main/java/com/yapp/mission/MissionContract.kt index 9aa81001..8e5c61d4 100644 --- a/feature/mission/src/main/java/com/yapp/mission/MissionContract.kt +++ b/feature/mission/src/main/java/com/yapp/mission/MissionContract.kt @@ -27,7 +27,6 @@ sealed class MissionContract { data object ClickCard : Action() data object ShowExitDialog : Action() data object HideExitDialog : Action() - data object RetryPostFortune : Action() } sealed class SideEffect : com.yapp.ui.base.SideEffect { diff --git a/feature/mission/src/main/java/com/yapp/mission/MissionNavGraph.kt b/feature/mission/src/main/java/com/yapp/mission/MissionNavGraph.kt index 247a9e5c..5befebf4 100644 --- a/feature/mission/src/main/java/com/yapp/mission/MissionNavGraph.kt +++ b/feature/mission/src/main/java/com/yapp/mission/MissionNavGraph.kt @@ -40,7 +40,7 @@ private fun handleSideEffect( MissionContract.SideEffect.NavigateToFortune -> { navigator.navigateToFortune( navOptions = navOptions { - popUpTo(MissionRoute.route) { + popUpTo { inclusive = true } }, @@ -50,7 +50,7 @@ private fun handleSideEffect( MissionContract.SideEffect.NavigateToHome -> { navigator.navigateToHome( navOptions = navOptions { - popUpTo(MissionRoute.route) { + popUpTo { inclusive = true } }, diff --git a/feature/mission/src/main/java/com/yapp/mission/MissionScreen.kt b/feature/mission/src/main/java/com/yapp/mission/MissionScreen.kt index ad178833..65249b8c 100644 --- a/feature/mission/src/main/java/com/yapp/mission/MissionScreen.kt +++ b/feature/mission/src/main/java/com/yapp/mission/MissionScreen.kt @@ -143,12 +143,6 @@ fun MissionScreen( MissionSuccessOverlay() } - state.errorMessage?.let { - ErrorDialog(message = it) { - eventDispatcher(MissionContract.Action.RetryPostFortune) - } - } - if (state.missionMode == MissionMode.PREVIEW) { val insets = WindowInsets.navigationBars.asPaddingValues() @@ -405,16 +399,6 @@ fun MissionSuccessOverlay() { } } -@Composable -fun ErrorDialog(message: String, onConfirm: () -> Unit) { - OrbitDialog( - title = stringResource(id = R.string.error), - message = message, - confirmText = stringResource(id = R.string.confirm), - onConfirm = onConfirm, - ) -} - @Composable fun MissionLoadingScreen() { Box( diff --git a/feature/mission/src/main/java/com/yapp/mission/MissionViewModel.kt b/feature/mission/src/main/java/com/yapp/mission/MissionViewModel.kt index 9d8808e4..b5e1c45d 100644 --- a/feature/mission/src/main/java/com/yapp/mission/MissionViewModel.kt +++ b/feature/mission/src/main/java/com/yapp/mission/MissionViewModel.kt @@ -7,16 +7,14 @@ import com.yapp.alarm.pendingIntent.interaction.createAlarmDismissIntent import com.yapp.analytics.AnalyticsEvent import com.yapp.analytics.AnalyticsHelper import com.yapp.domain.MissionMode +import com.yapp.domain.model.FortuneCreateStatus import com.yapp.domain.model.MissionType import com.yapp.domain.repository.FortuneRepository -import com.yapp.domain.repository.UserInfoRepository import com.yapp.media.haptic.HapticFeedbackManager import com.yapp.media.haptic.HapticType import dagger.hilt.android.lifecycle.HiltViewModel -import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.delay -import kotlinx.coroutines.flow.firstOrNull -import kotlinx.coroutines.withContext +import kotlinx.coroutines.flow.first import org.orbitmvi.orbit.Container import org.orbitmvi.orbit.ContainerHost import org.orbitmvi.orbit.syntax.simple.intent @@ -30,7 +28,6 @@ class MissionViewModel @Inject constructor( private val analyticsHelper: AnalyticsHelper, private val hapticFeedbackManager: HapticFeedbackManager, private val fortuneRepository: FortuneRepository, - private val userInfoRepository: UserInfoRepository, private val app: Application, private val savedStateHandle: SavedStateHandle, ) : ViewModel(), ContainerHost { @@ -53,7 +50,6 @@ class MissionViewModel @Inject constructor( is MissionContract.Action.ClickCard -> handleMissionProgress(MissionType.TAP) is MissionContract.Action.ShowExitDialog -> showExitDialog() is MissionContract.Action.HideExitDialog -> hideExitDialog() - is MissionContract.Action.RetryPostFortune -> retryPostFortune() } } @@ -137,36 +133,27 @@ class MissionViewModel @Inject constructor( private fun completeMission(type: String) = intent { performHapticSuccess() logMissionSuccess(type) - if (state.missionMode == MissionMode.REAL) { - postFortune() - } else { + + if (state.missionMode != MissionMode.REAL) { postSideEffect(MissionContract.SideEffect.NavigateBack) + return@intent } - } - private fun postFortune(isRetry: Boolean = false) = intent { - val userId = userInfoRepository.userIdFlow.firstOrNull() ?: return@intent + val fortuneCreateStatus = fortuneRepository.fortuneCreateStatusFlow.first() + val hasUnseenFortune = fortuneRepository.hasUnseenFortuneFlow.first() - val result = withContext(Dispatchers.IO) { - fortuneRepository.postFortune(userId) - } - - result.onSuccess { data -> - fortuneRepository.saveFortuneId(data.id) - fortuneRepository.saveFortuneScore(data.avgFortuneScore) + val shouldOpenFortune = ( + fortuneCreateStatus is FortuneCreateStatus.Creating || + fortuneCreateStatus is FortuneCreateStatus.Success && hasUnseenFortune + ) - postSideEffect(MissionContract.SideEffect.NavigateToFortune) - }.onFailure { error -> - if (isRetry) { - navigateToHome() + postSideEffect( + if (shouldOpenFortune) { + MissionContract.SideEffect.NavigateToFortune } else { - reduce { state.copy(errorMessage = error.message) } - } - } - } - - fun retryPostFortune() { - postFortune(isRetry = true) + MissionContract.SideEffect.NavigateBack + }, + ) } private fun logMissionSuccess(type: String) { @@ -183,8 +170,4 @@ class MissionViewModel @Inject constructor( private fun performHapticSuccess() { hapticFeedbackManager.performHapticFeedback(HapticType.SUCCESS) } - - private fun navigateToHome() = intent { - postSideEffect(MissionContract.SideEffect.NavigateToHome) - } } diff --git a/feature/onboarding/src/main/java/com/yapp/onboarding/OnboardingAccessScreen.kt b/feature/onboarding/src/main/java/com/yapp/onboarding/OnboardingAccessScreen.kt index abc83d5c..1485ed59 100644 --- a/feature/onboarding/src/main/java/com/yapp/onboarding/OnboardingAccessScreen.kt +++ b/feature/onboarding/src/main/java/com/yapp/onboarding/OnboardingAccessScreen.kt @@ -21,9 +21,7 @@ import androidx.compose.foundation.layout.fillMaxSize import androidx.compose.foundation.layout.fillMaxWidth import androidx.compose.foundation.layout.height import androidx.compose.foundation.layout.imePadding -import androidx.compose.foundation.layout.navigationBarsPadding import androidx.compose.foundation.layout.padding -import androidx.compose.foundation.layout.statusBarsPadding import androidx.compose.material3.Text import androidx.compose.runtime.Composable import androidx.compose.runtime.DisposableEffect @@ -48,7 +46,6 @@ import androidx.lifecycle.compose.LocalLifecycleOwner import androidx.lifecycle.compose.collectAsStateWithLifecycle import com.google.accompanist.permissions.ExperimentalPermissionsApi import com.google.accompanist.permissions.PermissionStatus -import com.google.accompanist.permissions.shouldShowRationale import com.yapp.analytics.AnalyticsEvent import com.yapp.analytics.LocalAnalyticsHelper import com.yapp.designsystem.theme.OrbitTheme @@ -238,8 +235,6 @@ fun OnboardingAccessScreen( modifier = Modifier .fillMaxSize() .background(OrbitTheme.colors.gray_900) - .statusBarsPadding() - .navigationBarsPadding() .imePadding(), ) { if (!hasRequestedPermission) { diff --git a/feature/onboarding/src/main/java/com/yapp/onboarding/OnboardingBirthdayScreen.kt b/feature/onboarding/src/main/java/com/yapp/onboarding/OnboardingBirthdayScreen.kt index 39868a89..ee9180dc 100644 --- a/feature/onboarding/src/main/java/com/yapp/onboarding/OnboardingBirthdayScreen.kt +++ b/feature/onboarding/src/main/java/com/yapp/onboarding/OnboardingBirthdayScreen.kt @@ -10,9 +10,7 @@ import androidx.compose.foundation.layout.fillMaxSize import androidx.compose.foundation.layout.fillMaxWidth import androidx.compose.foundation.layout.height import androidx.compose.foundation.layout.imePadding -import androidx.compose.foundation.layout.navigationBarsPadding import androidx.compose.foundation.layout.padding -import androidx.compose.foundation.layout.statusBarsPadding import androidx.compose.material3.Text import androidx.compose.runtime.Composable import androidx.compose.runtime.LaunchedEffect @@ -104,8 +102,6 @@ fun OnboardingBirthdayScreen( modifier = Modifier .fillMaxSize() .background(OrbitTheme.colors.gray_900) - .statusBarsPadding() - .navigationBarsPadding() .imePadding(), ) { OnBoardingTopAppBar( diff --git a/feature/onboarding/src/main/java/com/yapp/onboarding/OnboardingCompleteScreen2.kt b/feature/onboarding/src/main/java/com/yapp/onboarding/OnboardingCompleteScreen2.kt index f99e8771..f0e752e1 100644 --- a/feature/onboarding/src/main/java/com/yapp/onboarding/OnboardingCompleteScreen2.kt +++ b/feature/onboarding/src/main/java/com/yapp/onboarding/OnboardingCompleteScreen2.kt @@ -9,10 +9,8 @@ import androidx.compose.foundation.layout.fillMaxSize import androidx.compose.foundation.layout.fillMaxWidth import androidx.compose.foundation.layout.height import androidx.compose.foundation.layout.imePadding -import androidx.compose.foundation.layout.navigationBarsPadding import androidx.compose.foundation.layout.offset import androidx.compose.foundation.layout.padding -import androidx.compose.foundation.layout.statusBarsPadding import androidx.compose.material3.Text import androidx.compose.runtime.Composable import androidx.compose.ui.Alignment @@ -57,8 +55,6 @@ fun OnboardingCompleteScreen2( modifier = Modifier .fillMaxSize() .background(OrbitTheme.colors.gray_900) - .statusBarsPadding() - .navigationBarsPadding() .imePadding(), ) { OnBoardingTopAppBar( diff --git a/feature/onboarding/src/main/java/com/yapp/onboarding/OnboardingGenderScreen.kt b/feature/onboarding/src/main/java/com/yapp/onboarding/OnboardingGenderScreen.kt index 06a2ec94..01f0df00 100644 --- a/feature/onboarding/src/main/java/com/yapp/onboarding/OnboardingGenderScreen.kt +++ b/feature/onboarding/src/main/java/com/yapp/onboarding/OnboardingGenderScreen.kt @@ -29,6 +29,7 @@ import com.yapp.common.navigation.OrbitNavigator import com.yapp.common.navigation.route.OnboardingBaseRoute import com.yapp.designsystem.theme.OrbitTheme import com.yapp.onboarding.component.UserInfoBottomSheet +import com.yapp.ui.component.bottomsheet.OrbitBottomSheetLayout import com.yapp.ui.component.bottomsheet.OrbitBottomSheetState import com.yapp.ui.component.bottomsheet.rememberOrbitBottomSheetState import com.yapp.ui.component.dialog.OrbitDialog @@ -120,7 +121,7 @@ private suspend fun handleSideEffect( OnboardingContract.SideEffect.OnboardingCompleted -> { navigator.navigateToHome( navOptions = navOptions { - popUpTo(OnboardingBaseRoute) { + popUpTo { inclusive = true } }, @@ -153,55 +154,57 @@ fun OnboardingGenderScreen( } } - OnboardingScreen( - currentStep = currentStep, - totalSteps = totalSteps, - isButtonEnabled = state.selectedGender != null, - onNextClick = { - processAction(OnboardingContract.Action.ShowBottomSheet) - }, - onBackClick = { - processAction(OnboardingContract.Action.PreviousStep) - }, - buttonLabel = "다음", - ) { - Column( - modifier = Modifier.fillMaxSize(), - horizontalAlignment = Alignment.CenterHorizontally, + OrbitBottomSheetLayout(sheetState = bottomSheetState) { + OnboardingScreen( + currentStep = currentStep, + totalSteps = totalSteps, + isButtonEnabled = state.selectedGender != null, + onNextClick = { + processAction(OnboardingContract.Action.ShowBottomSheet) + }, + onBackClick = { + processAction(OnboardingContract.Action.PreviousStep) + }, + buttonLabel = "다음", ) { - Spacer(modifier = Modifier.heightForScreenPercentage(0.05f)) - Text( - text = stringResource(id = R.string.onboarding_step6_text_title), - style = OrbitTheme.typography.heading1SemiBold, - color = OrbitTheme.colors.white, - modifier = Modifier.fillMaxWidth(), - textAlign = TextAlign.Center, - ) - - Row( - modifier = Modifier - .fillMaxWidth() - .padding(horizontal = 38.dp) - .paddingForScreenPercentage(topPercentage = 0.11f), - horizontalArrangement = Arrangement.spacedBy(15.dp), + Column( + modifier = Modifier.fillMaxSize(), + horizontalAlignment = Alignment.CenterHorizontally, ) { - listOf("남성", "여성").forEach { gender -> - Box(modifier = Modifier.weight(1f)) { - OrbitGenderToggle( - label = gender, - isSelected = state.selectedGender == gender, - onToggle = { - logEvent( - AnalyticsEvent( - type = "onboarding_gender_select", - properties = mapOf( - AnalyticsEvent.OnboardingPropertiesKeys.GENDER to gender, + Spacer(modifier = Modifier.heightForScreenPercentage(0.05f)) + Text( + text = stringResource(id = R.string.onboarding_step6_text_title), + style = OrbitTheme.typography.heading1SemiBold, + color = OrbitTheme.colors.white, + modifier = Modifier.fillMaxWidth(), + textAlign = TextAlign.Center, + ) + + Row( + modifier = Modifier + .fillMaxWidth() + .padding(horizontal = 38.dp) + .paddingForScreenPercentage(topPercentage = 0.11f), + horizontalArrangement = Arrangement.spacedBy(15.dp), + ) { + listOf("남성", "여성").forEach { gender -> + Box(modifier = Modifier.weight(1f)) { + OrbitGenderToggle( + label = gender, + isSelected = state.selectedGender == gender, + onToggle = { + logEvent( + AnalyticsEvent( + type = "onboarding_gender_select", + properties = mapOf( + AnalyticsEvent.OnboardingPropertiesKeys.GENDER to gender, + ), ), - ), - ) - processAction(OnboardingContract.Action.UpdateGender(gender)) - }, - ) + ) + processAction(OnboardingContract.Action.UpdateGender(gender)) + }, + ) + } } } } diff --git a/feature/onboarding/src/main/java/com/yapp/onboarding/OnboardingNavGraph.kt b/feature/onboarding/src/main/java/com/yapp/onboarding/OnboardingNavGraph.kt index ff7417d7..a642d861 100644 --- a/feature/onboarding/src/main/java/com/yapp/onboarding/OnboardingNavGraph.kt +++ b/feature/onboarding/src/main/java/com/yapp/onboarding/OnboardingNavGraph.kt @@ -123,7 +123,7 @@ private fun handleOnboardingCommonSideEffect( OnboardingContract.SideEffect.OnboardingCompleted -> { navigator.navigateToHome( navOptions = navOptions { - popUpTo(OnboardingBaseRoute) { + popUpTo { inclusive = true } }, diff --git a/feature/onboarding/src/main/java/com/yapp/onboarding/OnboardingViewModel.kt b/feature/onboarding/src/main/java/com/yapp/onboarding/OnboardingViewModel.kt index 24f3ba73..13fc35eb 100644 --- a/feature/onboarding/src/main/java/com/yapp/onboarding/OnboardingViewModel.kt +++ b/feature/onboarding/src/main/java/com/yapp/onboarding/OnboardingViewModel.kt @@ -147,9 +147,7 @@ class OnboardingViewModel @Inject constructor( alarmUseCase.insertAlarm( alarm = newAlarm, - ).onSuccess { - postSideEffect(OnboardingContract.SideEffect.OnboardingCompleted) - }.onFailure { + ).onFailure { Log.e("OnboardingViewModel", "Failed to create alarm", it) } }.onFailure { diff --git a/gradle/libs.versions.toml b/gradle/libs.versions.toml index c10bb87c..2d1248b6 100644 --- a/gradle/libs.versions.toml +++ b/gradle/libs.versions.toml @@ -29,6 +29,7 @@ androidx-app-compat = "1.7.0" androidx-core = "1.15.0" androidx-datastore = "1.1.1" androidx-room = "2.7.2" +androidx-work = "2.10.3" androidx-lifecycle = "2.8.7" @@ -44,11 +45,12 @@ activity-compose = "1.9.3" ## Hilt hilt = "2.51.1" hilt-navigation-compose = "1.2.0" +hilt-work = "1.2.0" ## Third Party okhttp = "4.12.0" retrofit = "2.11.0" -coil = "2.4.0" +coil = "2.7.0" # Google Libraries Versions google-service = "4.4.2" @@ -102,6 +104,9 @@ androidx-room-compiler = { group = "androidx.room", name = "room-compiler", vers androidx-room-testing = { group = "androidx.room", name = "room-testing", version.ref = "androidx-room" } androidx-room-paging = { group = "androidx.room", name = "room-paging", version.ref = "androidx-room" } androidx-annotation = { group = "androidx.annotation", name = "annotation", version.ref = "annotation" } +androidx-hilt-compiler = { group = "androidx.hilt", name = "hilt-compiler", version.ref = "hilt-work" } +androidx-work-runtime = { group = "androidx.work", name = "work-runtime-ktx", version.ref = "androidx-work" } +androidx-work-testing = { group = "androidx.work", name = "work-testing", version.ref = "androidx-work" } ## Compose Libraries activity-compose = { group = "androidx.activity", name = "activity-compose", version.ref = "activity-compose" } @@ -129,6 +134,7 @@ hilt-android = { group = "com.google.dagger", name = "hilt-android", version.ref hilt-android-testing = { group = "com.google.dagger", name = "hilt-android-testing", version.ref = "hilt" } hilt-android-compiler = { group = "com.google.dagger", name = "hilt-android-compiler", version.ref = "hilt" } hilt-navigation-compose = { group = "androidx.hilt", name = "hilt-navigation-compose", version.ref = "hilt-navigation-compose" } +hilt-worker = { group = "androidx.hilt", name = "hilt-work", version.ref = "hilt-work" } # Orbit orbit-core = { group = "org.orbit-mvi", name = "orbit-core", version.ref = "orbit" } diff --git a/project.dot.png b/project.dot.png index 070d91b3..77a616a5 100644 Binary files a/project.dot.png and b/project.dot.png differ