From 3e289c297e523b22d8a70b782e590a68f0b9a13c Mon Sep 17 00:00:00 2001 From: soopeach Date: Thu, 2 Jan 2025 21:22:40 +0900 Subject: [PATCH] =?UTF-8?q?:sparkles:=20=EB=9F=AC=EB=8B=9D=20=EC=A2=85?= =?UTF-8?q?=EB=A3=8C=20=EC=8B=9C=20=EC=9A=94=EC=B2=AD/=EC=9D=91=EB=8B=B5?= =?UTF-8?q?=20=EA=B7=9C=EA=B2=A9=20=EB=B3=80=EA=B2=BD?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../screens/running/RunningScreen.kt | 14 ++- .../viewmodel/RunningViewModel.kt | 98 +++++++++++++------ 2 files changed, 78 insertions(+), 34 deletions(-) diff --git a/presentation/src/main/java/com/whyranoid/presentation/screens/running/RunningScreen.kt b/presentation/src/main/java/com/whyranoid/presentation/screens/running/RunningScreen.kt index 56f30c70..b310317b 100644 --- a/presentation/src/main/java/com/whyranoid/presentation/screens/running/RunningScreen.kt +++ b/presentation/src/main/java/com/whyranoid/presentation/screens/running/RunningScreen.kt @@ -87,6 +87,7 @@ import com.whyranoid.presentation.theme.WalkieTypography import com.whyranoid.presentation.util.dpToPx import com.whyranoid.presentation.util.toPace import com.whyranoid.presentation.util.toRunningTime +import com.whyranoid.presentation.viewmodel.RunningScreenSideEffect import com.whyranoid.presentation.viewmodel.RunningScreenState import com.whyranoid.presentation.viewmodel.RunningViewModel import com.whyranoid.presentation.viewmodel.RunningViewModel.Companion.MAP_MAX_ZOOM @@ -95,6 +96,7 @@ import com.whyranoid.runningdata.model.RunningFinishData import com.whyranoid.runningdata.model.RunningState import org.koin.androidx.compose.koinViewModel import org.orbitmvi.orbit.compose.collectAsState +import org.orbitmvi.orbit.compose.collectSideEffect @Composable fun RunningScreen( @@ -102,6 +104,16 @@ fun RunningScreen( startWorker: () -> Unit, ) { val viewModel = koinViewModel() + val state by viewModel.collectAsState() + + viewModel.collectSideEffect { + when (it) { + is RunningScreenSideEffect.CompleteChallenge -> { it + // todo: navigation elements + println("완료한 것 "+ it) + } + } + } LaunchedEffect(LocalLifecycleOwner.current) { viewModel.startWorker = startWorker @@ -110,8 +122,6 @@ fun RunningScreen( viewModel.onTrackingButtonClicked() } - val state by viewModel.collectAsState() - RunningContent( state, viewModel::startRunning, diff --git a/presentation/src/main/java/com/whyranoid/presentation/viewmodel/RunningViewModel.kt b/presentation/src/main/java/com/whyranoid/presentation/viewmodel/RunningViewModel.kt index f57994d2..b518013a 100644 --- a/presentation/src/main/java/com/whyranoid/presentation/viewmodel/RunningViewModel.kt +++ b/presentation/src/main/java/com/whyranoid/presentation/viewmodel/RunningViewModel.kt @@ -5,6 +5,7 @@ import android.net.Uri import android.util.Log import androidx.lifecycle.ViewModel import androidx.lifecycle.viewModelScope +import com.whyranoid.domain.model.running.CompletedRunning import com.whyranoid.domain.model.running.RunningData import com.whyranoid.domain.model.running.RunningHistory import com.whyranoid.domain.model.running.RunningPosition @@ -16,6 +17,7 @@ import com.whyranoid.domain.usecase.running.GetRunningFollowerUseCase import com.whyranoid.domain.usecase.running.RunningFinishUseCase import com.whyranoid.domain.usecase.running.RunningStartUseCase import com.whyranoid.domain.usecase.running.SendLikeUseCase +import com.whyranoid.domain.util.toFormattedTimeStamp import com.whyranoid.presentation.model.UiState import com.whyranoid.presentation.model.running.RunningFollower import com.whyranoid.presentation.model.running.RunningInfo @@ -24,14 +26,20 @@ import com.whyranoid.presentation.model.running.TrackingMode import com.whyranoid.runningdata.RunningDataManager import com.whyranoid.runningdata.model.RunningFinishData import com.whyranoid.runningdata.model.RunningState +import kotlinx.coroutines.delay import kotlinx.coroutines.flow.distinctUntilChanged import kotlinx.coroutines.launch import org.orbitmvi.orbit.ContainerHost 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 -sealed class RunningScreenSideEffect +sealed interface RunningScreenSideEffect { + data class CompleteChallenge( + val completedChallenges: List, + ) : RunningScreenSideEffect +} data class RunningScreenState( val runningState: UiState = UiState.Idle, @@ -45,6 +53,7 @@ data class RunningScreenState( val editState: UiState = UiState.Idle, val selectedImage: UiState = UiState.Idle, val savingState: UiState = UiState.Idle, + val historyId: UiState = UiState.Idle, ) class RunningViewModel( @@ -126,20 +135,18 @@ class RunningViewModel( } } } - viewModelScope.launch { + intent { runningDataManager.runningState.collect { runningState -> - intent { - reduce { - val runningInfo = - runningState.runningData.toWalikeRunningData().toRunningInfo() - state.copy( - runningState = UiState.Success(runningState), - runningInfoState = UiState.Success(runningInfo), - trackingModeState = UiState.Success( - state.trackingModeState.getDataOrNull() ?: TrackingMode.FOLLOW, - ), - ) - } + reduce { + val runningInfo = + runningState.runningData.toWalikeRunningData().toRunningInfo() + state.copy( + runningState = UiState.Success(runningState), + runningInfoState = UiState.Success(runningInfo), + trackingModeState = UiState.Success( + state.trackingModeState.getDataOrNull() ?: TrackingMode.FOLLOW, + ), + ) } } } @@ -152,7 +159,10 @@ class RunningViewModel( runningRepository.removeListener() intent { reduce { - state.copy(trackingModeState = UiState.Success(TrackingMode.FOLLOW)) + state.copy( + trackingModeState = UiState.Success(TrackingMode.FOLLOW), + historyId = UiState.Success(it), + ) } } }.onFailure { @@ -175,27 +185,48 @@ class RunningViewModel( } } - fun finishRunning() { - viewModelScope.launch { - runningFinishUseCase().onSuccess { - intent { - state.runningInfoState.getDataOrNull()?.let { - reduce { - state.copy(runningResultInfoState = UiState.Success(it)) - } - } + fun finishRunning() = intent { + + runningDataManager.finishRunning().onSuccess { runningFinishData -> + + reduce { + state.copy( + runningFinishState = UiState.Success(runningFinishData) + ) + } + + val runningInfo = state.runningInfoState.getDataOrNull() + + if (runningInfo != null) { + + reduce { + state.copy( + runningResultInfoState = UiState.Success(runningInfo), + ) } - runningDataManager.finishRunning().onSuccess { runningFinishData -> - intent { - reduce { - state.copy(runningFinishState = UiState.Success(runningFinishData)) - } - } + + val result = runningFinishUseCase( + state.historyId.getDataOrNull()?.toInt() ?: 0, + runningFinishData.runningHistory.finishedAt.toFormattedTimeStamp(), + runningFinishData.runningHistory.totalRunningTime, + runningFinishData.runningHistory.totalDistance, + state.runningInfoState.getDataOrNull()?.calories?.toInt() ?: 0, + state.runningInfoState.getDataOrNull()?.steps ?: 0, + ) + + result.onSuccess { completedIssues -> + + postSideEffect(RunningScreenSideEffect.CompleteChallenge(completedIssues)) + + }.onFailure { + Log.d("finishRunning Failure", it.message.toString()) } - }.onFailure { - Log.d("finishRunning Failure", it.message.toString()) } + + }.onFailure { + Log.d("finishRunning Failure", it.message.toString()) } + } fun onTrackingButtonClicked() { @@ -207,12 +238,15 @@ class RunningViewModel( TrackingMode.NONE -> { TrackingMode.NO_FOLLOW } + TrackingMode.NO_FOLLOW -> { TrackingMode.FOLLOW } + TrackingMode.FOLLOW -> { TrackingMode.NONE } + else -> { TrackingMode.FOLLOW }