Skip to content

Commit

Permalink
[FEATURE]#218 : FeatureOverview UI 구현 및 로직 적용
Browse files Browse the repository at this point in the history
  • Loading branch information
jeongjaino committed Jan 30, 2025
1 parent 326cc20 commit 6108df0
Show file tree
Hide file tree
Showing 5 changed files with 246 additions and 0 deletions.
158 changes: 158 additions & 0 deletions app/src/main/kotlin/com/bff/wespot/component/FeatureOverviewDialog.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,158 @@
package com.bff.wespot.component

import androidx.compose.foundation.layout.Arrangement
import androidx.compose.foundation.layout.Box
import androidx.compose.foundation.layout.Column
import androidx.compose.foundation.layout.PaddingValues
import androidx.compose.foundation.layout.Row
import androidx.compose.foundation.layout.fillMaxHeight
import androidx.compose.foundation.layout.fillMaxSize
import androidx.compose.foundation.layout.fillMaxWidth
import androidx.compose.foundation.layout.height
import androidx.compose.foundation.layout.padding
import androidx.compose.foundation.layout.width
import androidx.compose.foundation.rememberScrollState
import androidx.compose.foundation.verticalScroll
import androidx.compose.material3.ExperimentalMaterial3Api
import androidx.compose.material3.Scaffold
import androidx.compose.runtime.Composable
import androidx.compose.runtime.LaunchedEffect
import androidx.compose.runtime.getValue
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
import androidx.compose.ui.layout.ContentScale
import androidx.compose.ui.platform.LocalContext
import androidx.compose.ui.res.stringResource
import androidx.compose.ui.unit.dp
import androidx.compose.ui.window.Dialog
import androidx.compose.ui.window.DialogProperties
import androidx.hilt.navigation.compose.hiltViewModel
import coil.compose.AsyncImage
import coil.request.ImageRequest
import com.bff.wespot.R
import com.bff.wespot.analytic.TrackScreenViewEvent
import com.bff.wespot.designsystem.component.button.WSButton
import com.bff.wespot.designsystem.component.button.WSButtonType
import com.bff.wespot.designsystem.component.header.WSTopBar
import com.bff.wespot.model.notification.NotificationType
import com.bff.wespot.state.featureoverview.FeatureOverviewAction
import com.bff.wespot.state.featureoverview.FeatureOverviewSideEffect
import com.bff.wespot.ui.util.handleSideEffect
import com.bff.wespot.viewmodel.FeatureOverviewViewModel
import org.orbitmvi.orbit.compose.collectAsState
import org.orbitmvi.orbit.compose.collectSideEffect

@OptIn(ExperimentalMaterial3Api::class)
@Composable
fun FeatureOverviewDialog(
viewModel: FeatureOverviewViewModel = hiltViewModel(),
notificationType: NotificationType,
onDismissButtonClicked: () -> Unit,
onNavigateButtonClicked: (String) -> Unit,
) {
val scrollState = rememberScrollState()

val state by viewModel.collectAsState()
val action = viewModel::onAction

handleSideEffect(viewModel.sideEffect)

viewModel.collectSideEffect {
when (it) {
is FeatureOverviewSideEffect.NavigateScreen -> onNavigateButtonClicked(it.deepLink)
FeatureOverviewSideEffect.DismissDialog -> onDismissButtonClicked()
}
}

Dialog(
onDismissRequest = { },
properties = DialogProperties(usePlatformDefaultWidth = false),
) {
Scaffold(
topBar = {
WSTopBar(
title = state.ui.headerText.text,
canNavigateBack = true,
navigateUp = {
action(FeatureOverviewAction.OnDismissButtonClicked)
},
)
}
) { innerPadding ->
Column(
modifier = Modifier
.padding(innerPadding)
.verticalScroll(scrollState),
) {
AsyncImage(
modifier = Modifier
.let {
if (state.ui.overview.isFillMaxWidth()) {
it.fillMaxWidth()
} else {
it.width(state.ui.overview.width.dp)
}
if (state.ui.overview.isFillMaxHeight()) {
it.fillMaxHeight()
} else {
it.height(state.ui.overview.height.dp)
}
},
model = ImageRequest.Builder(LocalContext.current)
.data(state.ui.overview.url)
.error(com.bff.wespot.designsystem.R.drawable.default_image)
.fallback(com.bff.wespot.designsystem.R.drawable.default_image)
.placeholder(com.bff.wespot.designsystem.R.drawable.default_image)
.crossfade(true)
.build(),
contentScale = ContentScale.FillWidth,
contentDescription = stringResource(R.string.feature_overview_description),
)
}
}

Box(
modifier = Modifier.fillMaxSize(),
contentAlignment = Alignment.BottomCenter,
) {
Row(
modifier = Modifier
.fillMaxWidth()
.padding(horizontal = 20.dp, vertical = 12.dp),
horizontalArrangement = Arrangement.spacedBy(8.dp),
) {
Box(modifier = Modifier.weight(1f)) {
WSButton(
buttonType = WSButtonType.Secondary,
text = state.ui.dismissButton.text,
paddingValues = PaddingValues(0.dp),
onClick = { action(FeatureOverviewAction.OnDismissButtonClicked) },
content = { it() },
)
}

Box(modifier = Modifier.weight(1f)) {
WSButton(
buttonType = WSButtonType.Primary,
text = state.ui.navigateButton.text,
paddingValues = PaddingValues(0.dp),
onClick = {
action(
FeatureOverviewAction.OnNavigateButtonClicked(
state.ui.navigateButton.link,
),
)
},
content = { it() },
)
}
}
}
}

LaunchedEffect(Unit) {
action(FeatureOverviewAction.OnFeatureOverViewDialogShow(notificationType))
}

TrackScreenViewEvent(screenName = "feature_overview_screen")
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
package com.bff.wespot.state.featureoverview

import com.bff.wespot.model.notification.NotificationType

sealed interface FeatureOverviewAction {
data class OnFeatureOverViewDialogShow(
val notificationType: NotificationType,
): FeatureOverviewAction
data class OnNavigateButtonClicked(val deepLink: String) : FeatureOverviewAction
data object OnDismissButtonClicked : FeatureOverviewAction
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
package com.bff.wespot.state.featureoverview

sealed interface FeatureOverviewSideEffect {
data object DismissDialog : FeatureOverviewSideEffect
data class NavigateScreen(val deepLink: String) : FeatureOverviewSideEffect
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
package com.bff.wespot.state.featureoverview

import com.bff.wespot.model.dynamicui.FeatureOverview

data class FeatureOverviewUiState(
val ui: FeatureOverview = FeatureOverview(),
)
Original file line number Diff line number Diff line change
@@ -0,0 +1,64 @@
package com.bff.wespot.viewmodel

import androidx.lifecycle.viewModelScope
import com.bff.wespot.common.extension.onNetworkFailure
import com.bff.wespot.domain.repository.dynamicui.DynamicUiRepository
import com.bff.wespot.model.notification.NotificationType
import com.bff.wespot.state.featureoverview.FeatureOverviewAction
import com.bff.wespot.state.featureoverview.FeatureOverviewSideEffect
import com.bff.wespot.state.featureoverview.FeatureOverviewUiState
import com.bff.wespot.ui.base.BaseViewModel
import com.bff.wespot.ui.model.SideEffect.Companion.toSideEffect
import dagger.hilt.android.lifecycle.HiltViewModel
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
import timber.log.Timber
import javax.inject.Inject

@HiltViewModel
class FeatureOverviewViewModel @Inject constructor(
private val dynamicUiRepository: DynamicUiRepository,
): BaseViewModel(), ContainerHost<FeatureOverviewUiState, FeatureOverviewSideEffect> {
override val container = container<FeatureOverviewUiState, FeatureOverviewSideEffect>(
FeatureOverviewUiState()
)

fun onAction(action: FeatureOverviewAction) {
when (action) {
is FeatureOverviewAction.OnFeatureOverViewDialogShow -> {
handleDialogShow(type = action.notificationType)
}
is FeatureOverviewAction.OnNavigateButtonClicked -> handleNavigateButtonClicked(action.deepLink)
FeatureOverviewAction.OnDismissButtonClicked -> handleDismissButtonClicked()
}
}

private fun handleDialogShow(type: NotificationType) = intent {
viewModelScope.launch {
dynamicUiRepository.getFeatureOverview(type)
.onSuccess {
reduce {
state.copy(ui = it)
}
}
.onNetworkFailure {
postSideEffect(it.toSideEffect())
}
.onFailure {
Timber.e(it)
}
}
}

private fun handleDismissButtonClicked() = intent {
postSideEffect(FeatureOverviewSideEffect.DismissDialog)
}

private fun handleNavigateButtonClicked(deepLink: String) = intent {
postSideEffect(FeatureOverviewSideEffect.NavigateScreen(deepLink))
}
}

0 comments on commit 6108df0

Please sign in to comment.