diff --git a/app/build.gradle.kts b/app/build.gradle.kts
index d56d2894..6c350ede 100644
--- a/app/build.gradle.kts
+++ b/app/build.gradle.kts
@@ -8,6 +8,7 @@ plugins {
android {
namespace = "com.yapp.orbit"
+ compileSdk = 35
defaultConfig {
versionCode = 6
@@ -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..6eabb63b 100644
--- a/app/src/main/AndroidManifest.xml
+++ b/app/src/main/AndroidManifest.xml
@@ -46,9 +46,7 @@
-
+
@@ -80,5 +78,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/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/receivers/AlarmInteractionActivityReceiver.kt b/core/alarm/src/main/java/com/yapp/alarm/receivers/AlarmInteractionActivityReceiver.kt
index 39de3171..8d2389b7 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
@@ -11,10 +11,9 @@ 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
@@ -42,24 +41,38 @@ class AlarmInteractionActivityReceiver(private val activity: ComponentActivity)
missionCount != -1
)
- if (!hasValidMissionData) return
-
- CoroutineScope(Dispatchers.IO).launch {
- val fortuneDate = fortuneRepository.fortuneDateFlow.firstOrNull()
- val todayDate = LocalDate.now().format(DateTimeFormatter.ISO_DATE)
-
- val shouldLaunchMission = fortuneDate != todayDate
+ val pending = goAsync()
+ CoroutineScope(Dispatchers.Main).launch {
+ try {
+ if (!hasValidMissionData) {
+ val hasUnseenFortune = withContext(Dispatchers.IO) {
+ fortuneRepository.hasUnseenFortuneFlow.first()
+ }
+ 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)
+ }
+ }
+ return@launch
+ }
- if (shouldLaunchMission) {
- context?.let {
+ 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(it.packageName)
- }
- it.startActivity(missionIntent)
+ 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/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 96596729..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,10 +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.model.MissionType
-import com.yapp.domain.repository.FortuneRepository
import com.yapp.domain.usecase.AlarmUseCase
import com.yapp.media.sound.SoundPlayer
import dagger.hilt.android.AndroidEntryPoint
@@ -34,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
@@ -55,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())
@@ -82,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 {
@@ -124,16 +121,14 @@ class AlarmService : Service() {
if (isOneTimeAlarm) {
turnOffAlarm(alarmId = notificationId)
}
+
+ postFortuneTaskScheduler.enqueueOnceForToday()
}
- private suspend fun shouldNavigateToMission(
+ private fun shouldNavigateToMission(
missionType: MissionType,
): Boolean {
- if (missionType == MissionType.NONE) return false
-
- val fortuneDate = fortuneRepository.fortuneDateFlow.firstOrNull()
- val todayDate = LocalDate.now().format(DateTimeFormatter.ISO_DATE)
- return fortuneDate != todayDate
+ return missionType != MissionType.NONE
}
private fun createNotification(alarm: Alarm, shouldNavigateToMission: Boolean): Notification {
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..220dfa55 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
@@ -26,15 +25,22 @@ 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_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 = stringPreferencesKey("first_alarm_dismissed_date")
}
+ private fun today(): String = LocalDate.now().format(DateTimeFormatter.ISO_DATE)
+
val userIdFlow: Flow = dataStore.data
.catch { emit(emptyPreferences()) }
.map { it[Keys.USER_ID] }
@@ -70,108 +76,150 @@ 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 ->
+ pref[Keys.FORTUNE_DATE] == today() &&
+ 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] == today() && pref[Keys.FORTUNE_ID] != null
+ val tooltipNotShown = pref[Keys.FORTUNE_TOOLTIP_SHOWN] ?: false
+ hasTodayFortune && !tooltipNotShown
+ }
+ .distinctUntilChanged()
+
+ 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 date = pref[Keys.FIRST_ALARM_DISMISSED_DATE]
+ flag && date == today()
}
.distinctUntilChanged()
suspend fun saveUserId(userId: Long) {
- dataStore.edit { preferences ->
- preferences[Keys.USER_ID] = userId
+ dataStore.edit { pref ->
+ pref[Keys.USER_ID] = userId
}
}
suspend fun saveUserName(userName: String) {
- dataStore.edit { preferences ->
- preferences[Keys.USER_NAME] = userName
+ dataStore.edit { pref ->
+ pref[Keys.USER_NAME] = userName
}
}
- 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 setOnboardingCompleted() {
+ dataStore.edit { pref ->
+ pref[Keys.ONBOARDING_COMPLETED] = true
}
}
- suspend fun markFortuneAsChecked() {
- dataStore.edit { preferences ->
- preferences[Keys.FORTUNE_CHECKED] = true
+ suspend fun tryMarkFortuneCreating(): Boolean {
+ var canStart = false
+ dataStore.edit { pref ->
+ val hasTodayFortune = pref[Keys.FORTUNE_DATE] == today() && pref[Keys.FORTUNE_ID] != null
+ val creating = pref[Keys.FORTUNE_CREATING] ?: false
+ if (!hasTodayFortune && !creating) {
+ pref[Keys.FORTUNE_CREATING] = true
+ pref[Keys.FORTUNE_FAILED] = false
+ canStart = true
+ }
}
+ return canStart
}
- suspend fun saveFortuneImageId(imageResId: Int) {
- dataStore.edit { preferences ->
- preferences[Keys.FORTUNE_IMAGE_ID] = imageResId
+ suspend fun markFortuneCreating() {
+ dataStore.edit { pref ->
+ pref[Keys.FORTUNE_CREATING] = true
+ pref[Keys.FORTUNE_FAILED] = false
}
}
- suspend fun saveFortuneScore(score: Int) {
- dataStore.edit { preferences ->
- preferences[Keys.FORTUNE_SCORE] = score
+ suspend fun markFortuneCreated(fortuneId: Long) {
+ dataStore.edit { pref ->
+ val isNewForToday = pref[Keys.FORTUNE_ID] != fortuneId || pref[Keys.FORTUNE_DATE] != today()
+
+ pref[Keys.FORTUNE_ID] = fortuneId
+ pref[Keys.FORTUNE_DATE] = 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 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 markFortuneFailed() {
+ dataStore.edit { pref ->
+ pref[Keys.FORTUNE_CREATING] = false
+ pref[Keys.FORTUNE_FAILED] = true
}
}
- suspend fun setOnboardingCompleted() {
- dataStore.edit { preferences ->
- preferences[Keys.ONBOARDING_COMPLETED] = true
+ suspend fun markFortuneSeen() {
+ dataStore.edit { pref ->
+ pref[Keys.FORTUNE_SEEN] = true
}
}
- suspend fun clearDismissedAlarmId() {
- dataStore.edit { preferences ->
- preferences.remove(Keys.FIRST_DISMISSED_ALARM_ID)
- preferences.remove(Keys.DISMISSED_DATE)
+ suspend fun markFortuneTooltipShown() {
+ dataStore.edit { pref ->
+ pref[Keys.FORTUNE_TOOLTIP_SHOWN] = true
}
}
- suspend fun clearUserData() {
- dataStore.edit { preferences ->
- preferences.clear()
+ suspend fun saveFortuneImageId(imageResId: Int) {
+ dataStore.edit { pref ->
+ pref[Keys.FORTUNE_IMAGE_ID] = imageResId
}
}
- 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 saveFortuneScore(score: Int) {
+ dataStore.edit { pref ->
+ pref[Keys.FORTUNE_SCORE] = score
+ }
+ }
+
+ suspend fun markFirstAlarmDismissedToday() {
+ dataStore.edit { pref ->
+ pref[Keys.FIRST_ALARM_DISMISSED_TODAY] = true
+ pref[Keys.FIRST_ALARM_DISMISSED_DATE] = today()
+ }
+ }
+
+ suspend fun clearUserData() {
+ dataStore.edit { pref -> pref.clear() }
+ }
+
+ suspend fun clearFortuneData() {
+ dataStore.edit { pref ->
+ pref.remove(Keys.FORTUNE_ID)
+ pref.remove(Keys.FORTUNE_DATE)
+ 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/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..7b574bc0 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
@@ -7,14 +7,21 @@ interface FortuneLocalDataSource {
val fortuneDateFlow: Flow
val fortuneImageIdFlow: Flow
val fortuneScoreFlow: Flow
- val hasNewFortuneFlow: Flow
- val firstDismissedAlarmIdFlow: Flow
+ val hasUnseenFortuneFlow: Flow
+ val shouldShowFortuneToolTipFlow: Flow
+ val isFortuneCreatingFlow: Flow
+ val isFortuneFailedFlow: Flow
+ val isFirstAlarmDismissedTodayFlow: Flow
- suspend fun saveFortuneId(fortuneId: Long)
- suspend fun markFortuneAsChecked()
+ suspend fun tryMarkFortuneCreating(): Boolean
+ 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..80712f16 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
@@ -11,15 +11,34 @@ class FortuneLocalDataSourceImpl @Inject constructor(
override val fortuneDateFlow = userPreferences.fortuneDateFlow
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 isFortuneCreatingFlow = userPreferences.isFortuneCreatingFlow
+ override val isFortuneFailedFlow = userPreferences.isFortuneFailedFlow
+ override val isFirstAlarmDismissedTodayFlow = userPreferences.isFirstAlarmDismissedTodayFlow
+
+ override suspend fun tryMarkFortuneCreating(): Boolean {
+ return userPreferences.tryMarkFortuneCreating()
+ }
+
+ override suspend fun markFortuneCreating() {
+ userPreferences.markFortuneCreating()
+ }
+
+ override suspend fun markFortuneCreated(fortuneId: Long) {
+ userPreferences.markFortuneCreated(fortuneId)
+ }
- override suspend fun saveFortuneId(fortuneId: Long) {
- userPreferences.saveFortuneId(fortuneId)
+ override suspend fun markFortuneFailed() {
+ userPreferences.markFortuneFailed()
}
- override suspend fun markFortuneAsChecked() {
- userPreferences.markFortuneAsChecked()
+ override suspend fun markFortuneSeen() {
+ userPreferences.markFortuneSeen()
+ }
+
+ override suspend fun markFortuneTooltipShown() {
+ userPreferences.markFortuneTooltipShown()
}
override suspend fun saveFortuneImageId(imageResId: Int) {
@@ -30,15 +49,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/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..23271c5f 100644
--- a/data/src/main/java/com/yapp/data/repositoryimpl/FortuneRepositoryImpl.kt
+++ b/data/src/main/java/com/yapp/data/repositoryimpl/FortuneRepositoryImpl.kt
@@ -12,32 +12,36 @@ 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 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 isFortuneCreatingFlow: Flow = fortuneLocalDataSource.isFortuneCreatingFlow
+ override val isFortuneFailedFlow: Flow = fortuneLocalDataSource.isFortuneFailedFlow
+ override val isFirstAlarmDismissedTodayFlow: Flow = fortuneLocalDataSource.isFirstAlarmDismissedTodayFlow
- override suspend fun saveFortuneId(fortuneId: Long) = fortuneLocalDataSource.saveFortuneId(fortuneId)
- override suspend fun markFortuneAsChecked() = fortuneLocalDataSource.markFortuneAsChecked()
+ override suspend fun tryMarkFortuneCreating() = fortuneLocalDataSource.tryMarkFortuneCreating()
+ 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/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/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..89cc4a67 100644
--- a/domain/src/main/java/com/yapp/domain/repository/FortuneRepository.kt
+++ b/domain/src/main/java/com/yapp/domain/repository/FortuneRepository.kt
@@ -8,16 +8,24 @@ interface FortuneRepository {
val fortuneDateFlow: Flow
val fortuneImageIdFlow: Flow
val fortuneScoreFlow: Flow
- val hasNewFortuneFlow: Flow
- val firstDismissedAlarmIdFlow: Flow
+ val hasUnseenFortuneFlow: Flow
+ val shouldShowFortuneToolTipFlow: Flow
+ val isFortuneCreatingFlow: Flow
+ val isFortuneFailedFlow: Flow
+ val isFirstAlarmDismissedTodayFlow: Flow
- suspend fun saveFortuneId(fortuneId: Long)
- suspend fun markFortuneAsChecked()
+ suspend fun tryMarkFortuneCreating(): Boolean
+ 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/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 b4ca2c5c..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
@@ -7,10 +7,8 @@ import com.yapp.alarm.pendingIntent.interaction.createAlarmDismissIntent
import com.yapp.alarm.pendingIntent.interaction.createAlarmSnoozeIntent
import com.yapp.domain.model.Alarm
import com.yapp.domain.model.MissionType
-import com.yapp.domain.repository.FortuneRepository
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
@@ -19,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
@@ -27,7 +24,6 @@ import javax.inject.Inject
@HiltViewModel
class AlarmActionViewModel @Inject constructor(
private val app: Application,
- private val fortuneRepository: FortuneRepository,
savedStateHandle: SavedStateHandle,
) : ViewModel(), ContainerHost {
@@ -60,15 +56,8 @@ class AlarmActionViewModel @Inject constructor(
}
private fun fetchShouldShowMissionStart() = intent {
- val missionType = alarm?.missionType
- val fortuneDate = fortuneRepository.fortuneDateFlow.firstOrNull()
- val todayDate = LocalDate.now().format(DateTimeFormatter.ISO_DATE)
- val shouldShowMissionStart = missionType != null &&
- missionType != MissionType.NONE &&
- fortuneDate != todayDate
-
reduce {
- state.copy(shouldShowMissionStart = shouldShowMissionStart)
+ state.copy(shouldShowMissionStart = (alarm?.missionType ?: MissionType.NONE) != MissionType.NONE)
}
}
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..003aa09d 100644
--- a/feature/fortune/src/main/java/com/yapp/fortune/FortuneViewModel.kt
+++ b/feature/fortune/src/main/java/com/yapp/fortune/FortuneViewModel.kt
@@ -9,6 +9,7 @@ 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.combine
import kotlinx.coroutines.flow.firstOrNull
import org.orbitmvi.orbit.Container
import org.orbitmvi.orbit.ContainerHost
@@ -16,8 +17,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 +29,7 @@ class FortuneViewModel @Inject constructor(
override val container: Container = container(
initialState = FortuneContract.State(),
) {
- loadFortune()
+ observeFortune()
}
fun processAction(action: FortuneContract.Action) {
@@ -50,14 +49,32 @@ 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 {
+ combine(
+ fortuneRepository.fortuneIdFlow,
+ fortuneRepository.isFirstAlarmDismissedTodayFlow,
+ fortuneRepository.isFortuneCreatingFlow,
+ ) { fortuneId: Long?, isFirstAlarmDismissedToday: Boolean, isCreating: Boolean ->
+ Triple(fortuneId, isFirstAlarmDismissedToday, isCreating)
+ }.collect { (fortuneId, isFirstAlarmDismissedToday, isCreating) ->
+ when {
+ isCreating -> {
+ reduce { state.copy(isLoading = true) }
+ }
+ fortuneId != null -> {
+ fetchAndUpdateFortune(fortuneId, isFirstAlarmDismissedToday)
+ }
+ else -> {
+ reduce { state.copy(isLoading = false) }
+ }
+ }
+ }
}
- 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..751ae536
--- /dev/null
+++ b/feature/fortune/src/main/java/com/yapp/fortune/worker/PostFortuneWorker.kt
@@ -0,0 +1,54 @@
+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.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 {
+ val hasUnseenFortune = fortuneRepository.hasUnseenFortuneFlow.first()
+ if (hasUnseenFortune) return Result.success()
+
+ val acquired = fortuneRepository.tryMarkFortuneCreating()
+ if (!acquired) return Result.success()
+
+ 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 = { e ->
+ fortuneRepository.markFortuneAsFailed()
+ Result.retry()
+ },
+ )
+ } catch (_: Throwable) {
+ fortuneRepository.markFortuneAsFailed()
+ Result.retry()
+ }
+ }
+}
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..1834e476 100644
--- a/feature/home/src/main/java/com/yapp/home/HomeViewModel.kt
+++ b/feature/home/src/main/java/com/yapp/home/HomeViewModel.kt
@@ -338,7 +338,7 @@ class HomeViewModel @Inject constructor(
if (fortuneDate != todayDate) {
processAction(HomeContract.Action.ShowNoDailyFortuneDialog)
} else {
- fortuneRepository.markFortuneAsChecked()
+ fortuneRepository.markFortuneTooltipShown()
postSideEffect(HomeContract.SideEffect.NavigateToFortune)
}
}
@@ -349,12 +349,12 @@ class HomeViewModel @Inject constructor(
combine(
fortuneRepository.fortuneDateFlow,
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(
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/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..f54f58e4 100644
--- a/feature/mission/src/main/java/com/yapp/mission/MissionViewModel.kt
+++ b/feature/mission/src/main/java/com/yapp/mission/MissionViewModel.kt
@@ -9,14 +9,11 @@ import com.yapp.analytics.AnalyticsHelper
import com.yapp.domain.MissionMode
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 +27,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 +49,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()
}
}
@@ -138,37 +133,19 @@ class MissionViewModel @Inject constructor(
performHapticSuccess()
logMissionSuccess(type)
if (state.missionMode == MissionMode.REAL) {
- postFortune()
- } else {
- postSideEffect(MissionContract.SideEffect.NavigateBack)
- }
- }
-
- private fun postFortune(isRetry: Boolean = false) = intent {
- val userId = userInfoRepository.userIdFlow.firstOrNull() ?: return@intent
-
- val result = withContext(Dispatchers.IO) {
- fortuneRepository.postFortune(userId)
- }
+ val hasUnseenFortune = fortuneRepository.hasUnseenFortuneFlow.first()
+ val isFortuneCreating = fortuneRepository.isFortuneCreatingFlow.first()
- result.onSuccess { data ->
- fortuneRepository.saveFortuneId(data.id)
- fortuneRepository.saveFortuneScore(data.avgFortuneScore)
-
- postSideEffect(MissionContract.SideEffect.NavigateToFortune)
- }.onFailure { error ->
- if (isRetry) {
- navigateToHome()
+ if (hasUnseenFortune || isFortuneCreating) {
+ postSideEffect(MissionContract.SideEffect.NavigateToFortune)
} else {
- reduce { state.copy(errorMessage = error.message) }
+ postSideEffect(MissionContract.SideEffect.NavigateBack)
}
+ } else {
+ postSideEffect(MissionContract.SideEffect.NavigateBack)
}
}
- fun retryPostFortune() {
- postFortune(isRetry = true)
- }
-
private fun logMissionSuccess(type: String) {
analyticsHelper.logEvent(
AnalyticsEvent(
@@ -183,8 +160,4 @@ class MissionViewModel @Inject constructor(
private fun performHapticSuccess() {
hapticFeedbackManager.performHapticFeedback(HapticType.SUCCESS)
}
-
- private fun navigateToHome() = intent {
- postSideEffect(MissionContract.SideEffect.NavigateToHome)
- }
}
diff --git a/gradle/libs.versions.toml b/gradle/libs.versions.toml
index c10bb87c..db4817c9 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,6 +45,7 @@ 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"
@@ -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" }