From be7abcb866485f7ab57987107e892cd41502620d Mon Sep 17 00:00:00 2001 From: boiledegg Date: Sun, 1 Sep 2024 14:14:36 +0900 Subject: [PATCH 01/26] =?UTF-8?q?[MOD/#212]=20=EC=8A=A4=ED=81=AC=EB=9E=A9?= =?UTF-8?q?=20=EC=B7=A8=EC=86=8C=20=EB=8B=A4=EC=9D=B4=EC=96=BC=EB=A1=9C?= =?UTF-8?q?=EA=B7=B8=20=EC=B6=94=EC=B6=9C?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../src/main/assets/terning_scrap_cancel.json | 1 + .../calendar/week/CalendarWeekScreen.kt | 38 +++++- .../dialog/cancel/ScrapCancelDialog.kt | 108 ++++++++++++++++++ .../dialog/cancel/ScrapCancelSideEffect.kt | 6 + .../dialog/cancel/ScrapCancelViewModel.kt | 32 ++++++ 5 files changed, 184 insertions(+), 1 deletion(-) create mode 100644 feature/src/main/assets/terning_scrap_cancel.json create mode 100644 feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt create mode 100644 feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelSideEffect.kt create mode 100644 feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelViewModel.kt diff --git a/feature/src/main/assets/terning_scrap_cancel.json b/feature/src/main/assets/terning_scrap_cancel.json new file mode 100644 index 000000000..2c5b45f06 --- /dev/null +++ b/feature/src/main/assets/terning_scrap_cancel.json @@ -0,0 +1 @@ +{"v":"4.8.0","meta":{"g":"LottieFiles AE 3.5.7","a":"","k":"","d":"","tc":""},"fr":29.9700012207031,"ip":0,"op":66.0000026882351,"w":2600,"h":2030,"nm":"aos_scrap_cancle","ddd":0,"assets":[{"id":"image_0","w":982,"h":773,"u":"","p":"","e":1},{"id":"image_1","w":979,"h":772,"u":"","p":"","e":1},{"id":"image_2","w":983,"h":398,"u":"","p":"","e":1},{"id":"image_3","w":982,"h":762,"u":"","p":"","e":1},{"id":"image_4","w":2031,"h":2031,"u":"","p":"","e":1}],"layers":[{"ddd":0,"ind":1,"ty":2,"nm":"before","refId":"image_0","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":6,"s":[100]},{"t":8.00000032584668,"s":[0]}],"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[1],"y":[0.4]},"t":0,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":20,"s":[50]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":33,"s":[32]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":43,"s":[44]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":53,"s":[42]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":62,"s":[44]},{"t":65.0000026475043,"s":[44]}],"ix":10},"p":{"a":0,"k":[837.874,846.891,0],"ix":2},"a":{"a":0,"k":[28.577,28.519,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ip":0,"op":66.0000026882351,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":2,"nm":"after","refId":"image_1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.948]},"o":{"x":[1],"y":[0.4]},"t":0,"s":[0]},{"i":{"x":[0.833],"y":[0.978]},"o":{"x":[0.333],"y":[0]},"t":20,"s":[50]},{"i":{"x":[0.863],"y":[1.022]},"o":{"x":[0.333],"y":[0]},"t":33,"s":[34]},{"i":{"x":[0.671],"y":[1]},"o":{"x":[0.457],"y":[0]},"t":43,"s":[44]},{"i":{"x":[0.833],"y":[0.5]},"o":{"x":[0.333],"y":[0]},"t":53,"s":[41]},{"i":{"x":[0.833],"y":[0.5]},"o":{"x":[0.167],"y":[0]},"t":62,"s":[44]},{"t":65.0000026475043,"s":[44]}],"ix":10},"p":{"a":0,"k":[837.874,846.891,0],"ix":2},"a":{"a":0,"k":[28.577,28.519,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ip":0,"op":66.0000026882351,"st":0,"bm":0},{"ddd":0,"ind":3,"ty":2,"nm":"basic","refId":"image_2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[1300.427,647.931,0],"ix":2},"a":{"a":0,"k":[491.422,198.705,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ip":0,"op":66.0000026882351,"st":0,"bm":0},{"ddd":0,"ind":4,"ty":2,"nm":"back_cal","refId":"image_3","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[1301.426,1200.386,0],"ix":2},"a":{"a":0,"k":[490.883,380.927,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ip":0,"op":66.0000026882351,"st":0,"bm":0},{"ddd":0,"ind":5,"ty":2,"nm":"background","refId":"image_4","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[1300,1015.081,0],"ix":2},"a":{"a":0,"k":[1015.31,1015.31,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ip":0,"op":66.0000026882351,"st":0,"bm":0},{"ddd":0,"ind":6,"ty":1,"nm":"back","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[1300,1015,0],"ix":2},"a":{"a":0,"k":[1300,1015,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"sw":2600,"sh":2030,"sc":"#ffffff","ip":0,"op":66.0000026882351,"st":0,"bm":0}],"markers":[]} \ No newline at end of file diff --git a/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekScreen.kt b/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekScreen.kt index 44ee9d65c..ca919b3f8 100644 --- a/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekScreen.kt +++ b/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekScreen.kt @@ -12,10 +12,15 @@ import androidx.compose.material3.Text import androidx.compose.runtime.Composable import androidx.compose.runtime.LaunchedEffect import androidx.compose.runtime.getValue +import androidx.compose.runtime.mutableStateOf +import androidx.compose.runtime.remember +import androidx.compose.runtime.setValue import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.draw.shadow +import androidx.compose.ui.geometry.Offset import androidx.compose.ui.graphics.Color +import androidx.compose.ui.input.pointer.pointerInput import androidx.compose.ui.platform.LocalContext import androidx.compose.ui.res.stringResource import androidx.compose.ui.text.style.TextAlign @@ -110,12 +115,43 @@ private fun CalendarWeekScreen( navigateToAnnouncement: (Long) -> Unit, modifier: Modifier = Modifier ) { + var initialTouchPosition by remember { mutableStateOf(null) } + var hideComponent by remember { mutableStateOf(false) } + + LaunchedEffect(hideComponent) { + if(hideComponent) { + updateSelectedDate(selectedDate) + } + } + + val swipeModifier = Modifier.pointerInput(Unit) { + awaitPointerEventScope { + while (true) { + val event = awaitPointerEvent() + val position = event.changes.first().position + + if (event.changes.first().pressed) { + if (initialTouchPosition == null) { + initialTouchPosition = position + } else { + val deltaY = initialTouchPosition?.let { position.y - it.y } + if (deltaY != null && deltaY > 300f) { + hideComponent = true + } + } + } else { + initialTouchPosition = null + } + } + } + } + Column( modifier = modifier .background(Back) ) { Card( - modifier = Modifier + modifier = swipeModifier .shadow( shape = RoundedCornerShape(bottomStart = 20.dp, bottomEnd = 20.dp), elevation = 1.dp diff --git a/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt new file mode 100644 index 000000000..5e88c2de8 --- /dev/null +++ b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt @@ -0,0 +1,108 @@ +package com.terning.feature.dialog.cancel + +import androidx.compose.foundation.layout.Box +import androidx.compose.foundation.layout.Column +import androidx.compose.foundation.layout.aspectRatio +import androidx.compose.foundation.layout.fillMaxWidth +import androidx.compose.foundation.layout.padding +import androidx.compose.material3.Text +import androidx.compose.runtime.Composable +import androidx.compose.runtime.LaunchedEffect +import androidx.compose.ui.Alignment +import androidx.compose.ui.Modifier +import androidx.compose.ui.res.stringResource +import androidx.compose.ui.text.style.TextAlign +import androidx.compose.ui.tooling.preview.Preview +import androidx.compose.ui.unit.dp +import androidx.hilt.navigation.compose.hiltViewModel +import androidx.lifecycle.compose.LocalLifecycleOwner +import androidx.lifecycle.flowWithLifecycle +import com.terning.core.R +import com.terning.core.designsystem.component.button.RoundButton +import com.terning.core.designsystem.component.dialog.TerningBasicDialog +import com.terning.core.designsystem.component.item.TerningLottieAnimation +import com.terning.core.designsystem.theme.Grey350 +import com.terning.core.designsystem.theme.Grey500 +import com.terning.core.designsystem.theme.TerningPointTheme +import com.terning.core.designsystem.theme.TerningTheme + +@Composable +fun ScrapCancelDialog( + scrapId: Long, + onDismissRequest: (Boolean) -> Unit, + viewModel: ScrapCancelViewModel = hiltViewModel() +) { + val lifecycleOwner = LocalLifecycleOwner.current + LaunchedEffect(viewModel.sideEffect, lifecycleOwner) { + viewModel.sideEffect.flowWithLifecycle(lifecycle = lifecycleOwner.lifecycle) + .collect{ sideEffect -> + when(sideEffect){ + is ScrapCancelSideEffect.DismissDialog -> { + onDismissRequest(true) + } + is ScrapCancelSideEffect.ShowToast -> {} + } + } + } + + + TerningBasicDialog( + onDismissRequest = { onDismissRequest(false) } + ) { + Box( + modifier = Modifier + .fillMaxWidth() + .padding(top = 50.dp), + contentAlignment = Alignment.TopCenter + ) { + Column( + modifier = Modifier + .fillMaxWidth() + .padding(horizontal = 16.dp), + horizontalAlignment = Alignment.CenterHorizontally + ) { + TerningLottieAnimation( + jsonString = "terning_scrap_cancel.json", + modifier = Modifier + .fillMaxWidth() + .aspectRatio(1f) + ) + + Text( + text = stringResource(id = R.string.dialog_content_scrap_cancel_main_title), + textAlign = TextAlign.Center, + style = TerningTheme.typography.title4, + color = Grey500, + ) + Text( + text = stringResource(id = R.string.dialog_content_scrap_cancel_sub_title), + style = TerningTheme.typography.body5, + color = Grey350, + modifier = Modifier.padding( + top = 5.dp + ) + ) + RoundButton( + style = TerningTheme.typography.button3, + paddingVertical = 12.dp, + cornerRadius = 8.dp, + text = R.string.dialog_scrap_cancel_button, + onButtonClick = { viewModel.deleteScrap(scrapId) }, + modifier = Modifier.padding(bottom = 8.dp, top = 40.dp) + ) + } + } + } +} + + +@Preview(showBackground = true) +@Composable +private fun TerningBasicDialogPreview() { + TerningPointTheme { + ScrapCancelDialog( + scrapId = 1, + onDismissRequest = {} + ) + } +} \ No newline at end of file diff --git a/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelSideEffect.kt b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelSideEffect.kt new file mode 100644 index 000000000..c91071090 --- /dev/null +++ b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelSideEffect.kt @@ -0,0 +1,6 @@ +package com.terning.feature.dialog.cancel + +sealed class ScrapCancelSideEffect{ + data class ShowToast(val message: Int): ScrapCancelSideEffect() + data object DismissDialog : ScrapCancelSideEffect() +} \ No newline at end of file diff --git a/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelViewModel.kt b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelViewModel.kt new file mode 100644 index 000000000..5d3c6db22 --- /dev/null +++ b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelViewModel.kt @@ -0,0 +1,32 @@ +package com.terning.feature.dialog.cancel + +import androidx.lifecycle.ViewModel +import androidx.lifecycle.viewModelScope +import com.terning.domain.entity.CalendarScrapRequest +import com.terning.domain.repository.ScrapRepository +import com.terning.feature.R +import com.terning.feature.calendar.list.CalendarListSideEffect +import dagger.hilt.android.lifecycle.HiltViewModel +import kotlinx.coroutines.flow.MutableSharedFlow +import kotlinx.coroutines.flow.asSharedFlow +import kotlinx.coroutines.launch +import javax.inject.Inject + +@HiltViewModel +class ScrapCancelViewModel @Inject constructor( + private val scrapRepository: ScrapRepository +) : ViewModel() { + private var _sideEffect: MutableSharedFlow = MutableSharedFlow() + val sideEffect = _sideEffect.asSharedFlow() + + fun deleteScrap( + scrapId: Long + ) = viewModelScope.launch { + scrapRepository.deleteScrap(CalendarScrapRequest(scrapId, null)) + .onSuccess { + _sideEffect.emit(ScrapCancelSideEffect.DismissDialog) + }.onFailure { + _sideEffect.emit(ScrapCancelSideEffect.ShowToast(R.string.server_failure)) + } + } +} \ No newline at end of file From 7a0fa35217a41d125eba23ababe018bb400ee34b Mon Sep 17 00:00:00 2001 From: boiledegg Date: Sun, 1 Sep 2024 14:54:07 +0900 Subject: [PATCH 02/26] =?UTF-8?q?[FEAT/#212]=20=EC=8A=A4=ED=81=AC=EB=9E=A9?= =?UTF-8?q?=20=EC=B7=A8=EC=86=8C=EC=99=80=20=EC=BA=98=EB=A6=B0=EB=8D=94=20?= =?UTF-8?q?=EC=97=B0=EB=8F=99?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../calendar/list/CalendarListScreen.kt | 36 ++++++++++--------- .../calendar/week/CalendarWeekScreen.kt | 29 ++++++++------- .../dialog/cancel/ScrapCancelDialog.kt | 8 +++++ 3 files changed, 45 insertions(+), 28 deletions(-) diff --git a/feature/src/main/java/com/terning/feature/calendar/list/CalendarListScreen.kt b/feature/src/main/java/com/terning/feature/calendar/list/CalendarListScreen.kt index d5563b7ff..e7184ccaa 100644 --- a/feature/src/main/java/com/terning/feature/calendar/list/CalendarListScreen.kt +++ b/feature/src/main/java/com/terning/feature/calendar/list/CalendarListScreen.kt @@ -39,12 +39,12 @@ import com.terning.core.extension.toast import com.terning.core.state.UiState import com.terning.domain.entity.CalendarScrapDetail import com.terning.feature.R -import com.terning.feature.calendar.calendar.component.CalendarCancelDialog import com.terning.feature.calendar.calendar.component.CalendarDetailDialog import com.terning.feature.calendar.calendar.model.CalendarDefaults.flingBehavior import com.terning.feature.calendar.calendar.model.CalendarModel.Companion.getLocalDateByPage import com.terning.feature.calendar.list.component.CalendarScrapList import com.terning.feature.calendar.list.model.CalendarListUiState +import com.terning.feature.dialog.cancel.ScrapCancelDialog import kotlinx.coroutines.flow.distinctUntilChanged import java.time.LocalDate @@ -86,17 +86,23 @@ fun CalendarListRoute( uiState = uiState, modifier = modifier, navigateToAnnouncement = navigateToAnnouncement, - onDismissCancelDialog = { viewModel.updateScrapCancelDialogVisibility(false) }, + onDismissCancelDialog = { isCancelled -> + viewModel.updateScrapCancelDialogVisibility(false) + if (isCancelled) { viewModel.getScrapMonthList(uiState.currentDate) } + }, onDismissInternDialog = { viewModel.updateInternDialogVisibility(false) }, onClickChangeColor = { newColor -> viewModel.patchScrap(newColor) }, - onClickScrapCancel = { uiState.scrapId?.let { viewModel.deleteScrap(it) } }, onClickScrapButton = { scrapId -> - viewModel.updateScrapId(scrapId) - viewModel.updateScrapCancelDialogVisibility(true) + with(viewModel){ + updateScrapId(scrapId) + updateScrapCancelDialogVisibility(true) + } }, onClickInternship = { calendarScrapDetail -> - viewModel.updateInternshipModel(calendarScrapDetail) - viewModel.updateInternDialogVisibility(true) + with(viewModel) { + updateInternshipModel(calendarScrapDetail) + updateInternDialogVisibility(true) + } } ) } @@ -107,10 +113,9 @@ private fun CalendarListScreen( listState: LazyListState, uiState: CalendarListUiState, navigateToAnnouncement: (Long) -> Unit, - onDismissCancelDialog: () -> Unit, + onDismissCancelDialog: (Boolean) -> Unit, onDismissInternDialog: () -> Unit, onClickChangeColor: (Color) -> Unit, - onClickScrapCancel: () -> Unit, onClickInternship: (CalendarScrapDetail) -> Unit, onClickScrapButton: (Long) -> Unit, modifier: Modifier = Modifier @@ -189,13 +194,12 @@ private fun CalendarListScreen( } if (uiState.scrapDialogVisibility) { - CalendarCancelDialog( - onDismissRequest = onDismissCancelDialog, - onClickScrapCancel = { - onClickScrapCancel() - onDismissCancelDialog() - } - ) + uiState.scrapId?.run { + ScrapCancelDialog( + scrapId = this, + onDismissRequest = onDismissCancelDialog + ) + } } if (uiState.internDialogVisibility) { diff --git a/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekScreen.kt b/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekScreen.kt index ca919b3f8..2ccb983d4 100644 --- a/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekScreen.kt +++ b/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekScreen.kt @@ -38,12 +38,12 @@ import com.terning.core.extension.toast import com.terning.core.state.UiState import com.terning.domain.entity.CalendarScrapDetail import com.terning.feature.R -import com.terning.feature.calendar.calendar.component.CalendarCancelDialog import com.terning.feature.calendar.calendar.component.CalendarDetailDialog import com.terning.feature.calendar.calendar.model.CalendarUiState import com.terning.feature.calendar.list.component.CalendarScrapList import com.terning.feature.calendar.week.component.HorizontalCalendarWeek import com.terning.feature.calendar.week.model.CalendarWeekUiState +import com.terning.feature.dialog.cancel.ScrapCancelDialog import okhttp3.internal.toImmutableList import java.time.LocalDate @@ -82,11 +82,16 @@ fun CalendarWeekRoute( selectedDate = calendarUiState.selectedDate, updateSelectedDate = updateSelectedDate, navigateToAnnouncement = navigateToAnnouncement, - onDismissCancelDialog = { viewModel.updateScrapCancelDialogVisibility(false) }, + onDismissCancelDialog = { isCancelled -> + viewModel.updateScrapCancelDialogVisibility(false) + if (isCancelled) { + viewModel.getScrapWeekList(uiState.selectedDate) + } + }, onDismissInternDialog = { viewModel.updateInternDialogVisibility(false) }, onClickChangeColor = { viewModel.patchScrap(it) }, onClickScrapCancel = { uiState.scrapId?.let { viewModel.deleteScrap(it) } }, - onClickScrapButton = {scrapId -> + onClickScrapButton = { scrapId -> with(viewModel) { updateScrapId(scrapId) updateScrapCancelDialogVisibility(true) @@ -106,7 +111,7 @@ private fun CalendarWeekScreen( uiState: CalendarWeekUiState, selectedDate: LocalDate, updateSelectedDate: (LocalDate) -> Unit, - onDismissCancelDialog: () -> Unit, + onDismissCancelDialog: (Boolean) -> Unit, onDismissInternDialog: () -> Unit, onClickChangeColor: (Color) -> Unit, onClickScrapCancel: () -> Unit, @@ -119,7 +124,7 @@ private fun CalendarWeekScreen( var hideComponent by remember { mutableStateOf(false) } LaunchedEffect(hideComponent) { - if(hideComponent) { + if (hideComponent) { updateSelectedDate(selectedDate) } } @@ -173,6 +178,7 @@ private fun CalendarWeekScreen( is UiState.Empty -> { CalendarWeekEmpty() } + is UiState.Failure -> {} is UiState.Success -> { CalendarWeekSuccess( @@ -186,13 +192,12 @@ private fun CalendarWeekScreen( } if (uiState.scrapDialogVisibility) { - CalendarCancelDialog( - onDismissRequest = onDismissCancelDialog, - onClickScrapCancel = { - onClickScrapCancel() - onDismissCancelDialog() - } - ) + uiState.scrapId?.run { + ScrapCancelDialog( + scrapId = this, + onDismissRequest = onDismissCancelDialog + ) + } } if (uiState.internDialogVisibility) { diff --git a/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt index 5e88c2de8..1e97014c6 100644 --- a/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt +++ b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt @@ -26,6 +26,14 @@ import com.terning.core.designsystem.theme.Grey500 import com.terning.core.designsystem.theme.TerningPointTheme import com.terning.core.designsystem.theme.TerningTheme +/** + * 스크랩 취소 다이얼로그입니다 + * + * @param scrapId 공고의 스크랩 ID, 공고 ID로 변경될 예정입니다. + * @param onDismissRequest 다이얼로그를 닫을 때 호출되는 콜백 함수입니다. 스크랩이 취소되면 true를 반환합니다. + * @param viewModel 스크랩 취소 API 요청을 처리하는 ViewModel입니다. + */ + @Composable fun ScrapCancelDialog( scrapId: Long, From 5f95377c0e5b6e8ebc62d4c134ac92f7f38dadd7 Mon Sep 17 00:00:00 2001 From: boiledegg Date: Sun, 1 Sep 2024 15:00:00 +0900 Subject: [PATCH 03/26] =?UTF-8?q?[UI/#212]=20=EC=8A=A4=ED=81=AC=EB=9E=A9?= =?UTF-8?q?=20=EC=B7=A8=EC=86=8C=20=ED=94=84=EB=A6=AC=EB=B7=B0?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../dialog/cancel/ScrapCancelDialog.kt | 19 +++++++++++++++---- 1 file changed, 15 insertions(+), 4 deletions(-) diff --git a/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt index 1e97014c6..34ffb62c4 100644 --- a/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt +++ b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt @@ -53,9 +53,20 @@ fun ScrapCancelDialog( } } + ScrapCancelScreen( + onDismissRequest = { onDismissRequest(false) }, + onClickScrapCancel = { viewModel.deleteScrap(scrapId) } + ) +} + +@Composable +private fun ScrapCancelScreen( + onDismissRequest: () -> Unit, + onClickScrapCancel: () -> Unit, +) { TerningBasicDialog( - onDismissRequest = { onDismissRequest(false) } + onDismissRequest = onDismissRequest ) { Box( modifier = Modifier @@ -95,7 +106,7 @@ fun ScrapCancelDialog( paddingVertical = 12.dp, cornerRadius = 8.dp, text = R.string.dialog_scrap_cancel_button, - onButtonClick = { viewModel.deleteScrap(scrapId) }, + onButtonClick = onClickScrapCancel, modifier = Modifier.padding(bottom = 8.dp, top = 40.dp) ) } @@ -108,8 +119,8 @@ fun ScrapCancelDialog( @Composable private fun TerningBasicDialogPreview() { TerningPointTheme { - ScrapCancelDialog( - scrapId = 1, + ScrapCancelScreen( + onClickScrapCancel = {}, onDismissRequest = {} ) } From 337c03c5187ac9d473f17962d58da3c9a661826f Mon Sep 17 00:00:00 2001 From: boiledegg Date: Sun, 1 Sep 2024 17:15:22 +0900 Subject: [PATCH 04/26] =?UTF-8?q?[MOD=20[MOD/#212]=20=EA=B3=B5=EA=B3=A0=20?= =?UTF-8?q?=EB=8B=A4=EC=9D=B4=EC=96=BC=EB=A1=9C=EA=B7=B8=20=EC=B6=94?= =?UTF-8?q?=EC=B6=9C?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../dialog/cancel/ScrapCancelDialog.kt | 8 - .../feature/dialog/detail/ScrapDialog.kt | 334 ++++++++++++++++++ .../dialog/detail/ScrapDialogSideEffect.kt | 9 + .../dialog/detail/ScrapDialogViewModel.kt | 74 ++++ 4 files changed, 417 insertions(+), 8 deletions(-) create mode 100644 feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt create mode 100644 feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogSideEffect.kt create mode 100644 feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogViewModel.kt diff --git a/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt index 34ffb62c4..328656e75 100644 --- a/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt +++ b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt @@ -26,14 +26,6 @@ import com.terning.core.designsystem.theme.Grey500 import com.terning.core.designsystem.theme.TerningPointTheme import com.terning.core.designsystem.theme.TerningTheme -/** - * 스크랩 취소 다이얼로그입니다 - * - * @param scrapId 공고의 스크랩 ID, 공고 ID로 변경될 예정입니다. - * @param onDismissRequest 다이얼로그를 닫을 때 호출되는 콜백 함수입니다. 스크랩이 취소되면 true를 반환합니다. - * @param viewModel 스크랩 취소 API 요청을 처리하는 ViewModel입니다. - */ - @Composable fun ScrapCancelDialog( scrapId: Long, diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt new file mode 100644 index 000000000..30ab99189 --- /dev/null +++ b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt @@ -0,0 +1,334 @@ +package com.terning.feature.dialog.detail + +import androidx.compose.foundation.background +import androidx.compose.foundation.border +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.Spacer +import androidx.compose.foundation.layout.aspectRatio +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.layout.wrapContentSize +import androidx.compose.foundation.shape.RoundedCornerShape +import androidx.compose.material3.HorizontalDivider +import androidx.compose.material3.Text +import androidx.compose.runtime.Composable +import androidx.compose.runtime.LaunchedEffect +import androidx.compose.runtime.getValue +import androidx.compose.runtime.mutableStateOf +import androidx.compose.runtime.remember +import androidx.compose.runtime.setValue +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.res.stringResource +import androidx.compose.ui.text.style.TextAlign +import androidx.compose.ui.tooling.preview.Preview +import androidx.compose.ui.unit.dp +import androidx.hilt.navigation.compose.hiltViewModel +import androidx.lifecycle.compose.LocalLifecycleOwner +import androidx.lifecycle.flowWithLifecycle +import coil.compose.AsyncImage +import coil.request.ImageRequest +import com.terning.core.R +import com.terning.core.designsystem.component.button.RoundButton +import com.terning.core.designsystem.component.dialog.TerningBasicDialog +import com.terning.core.designsystem.component.item.ColorPalette +import com.terning.core.designsystem.theme.Grey100 +import com.terning.core.designsystem.theme.Grey200 +import com.terning.core.designsystem.theme.Grey350 +import com.terning.core.designsystem.theme.Grey400 +import com.terning.core.designsystem.theme.Grey500 +import com.terning.core.designsystem.theme.TerningMain +import com.terning.core.designsystem.theme.TerningPointTheme +import com.terning.core.designsystem.theme.TerningTheme +import com.terning.feature.intern.component.InternInfoRow + + +@Composable +fun ScrapDialog( + title: String, + scrapColor: Color, + deadline: String, + startYear: Int, + startMonth: Int, + workingPeriod: String, + scrapId: Long, + internshipAnnouncementId: Long, + companyImage: String, + isScrapped: Boolean, + onDismissRequest: () -> Unit, + onClickChangeColor: () -> Unit, + onClickNavigateButton: (Long) -> Unit, + viewModel: ScrapDialogViewModel = hiltViewModel() +) { + val lifecycleOwner = LocalLifecycleOwner.current + LaunchedEffect(viewModel.sideEffect, lifecycleOwner) { + viewModel.sideEffect.flowWithLifecycle(lifecycle = lifecycleOwner.lifecycle) + .collect { sideEffect -> + when (sideEffect) { + is ScrapDialogSideEffect.ShowToast -> {} + is ScrapDialogSideEffect.DismissDialog -> onDismissRequest() + is ScrapDialogSideEffect.ChangedColor -> onClickChangeColor() + is ScrapDialogSideEffect.NavigateToDetail -> onClickNavigateButton( + internshipAnnouncementId + ) + is ScrapDialogSideEffect.ScrappedAnnouncement -> {} + } + } + } + + TerningBasicDialog( + onDismissRequest = onDismissRequest + ) { + ScrapDialogScreen( + title = title, + scrapColor = scrapColor, + deadline = deadline, + startYear = startYear, + startMonth = startMonth, + workingPeriod = workingPeriod, + isScrapped = isScrapped, + companyImage = companyImage, + onClickColorChangeButton = { newColor -> + onClickChangeColor() + viewModel.patchScrap(scrapId = scrapId, color = newColor) + }, + onClickNavigateButton = viewModel::navigateToDetail, + onClickScrapButton = { viewModel.postScrap(internshipAnnouncementId, scrapColor) } + ) + } +} + + +@Composable +private fun ScrapDialogScreen( + title: String, + scrapColor: Color, + deadline: String, + startYear: Int, + startMonth: Int, + workingPeriod: String, + isScrapped: Boolean, + companyImage: String, + onClickNavigateButton: () -> Unit, + onClickColorChangeButton: (Color) -> Unit, + onClickScrapButton: () -> Unit +) { + var selectedColor by remember { mutableStateOf(scrapColor) } + + Box( + modifier = Modifier + .wrapContentSize() + .padding(top = 32.dp, bottom = 16.dp), + contentAlignment = Alignment.TopCenter + ) { + Column( + modifier = Modifier + .padding(horizontal = 11.dp), + horizontalAlignment = Alignment.CenterHorizontally + ) { + AsyncImage( + model = ImageRequest.Builder(LocalContext.current) + .data(companyImage) + .build(), + contentDescription = title, + contentScale = ContentScale.Fit, + modifier = Modifier + .width(80.dp) + .aspectRatio(1f) + .clip(RoundedCornerShape(15.dp)) + .border( + width = 1.dp, + color = TerningMain, + shape = RoundedCornerShape(size = 15.dp) + ) + ) + Text( + text = title, + textAlign = TextAlign.Center, + style = TerningTheme.typography.title4, + color = Grey500, + modifier = Modifier.padding(top = 20.dp) + ) + Text( + text = stringResource(id = R.string.dialog_scrap_mine), + style = TerningTheme.typography.body5, + color = Grey350, + modifier = Modifier.padding( + top = 4.dp + ) + ) + Spacer(modifier = Modifier.height(26.dp)) + Column( + horizontalAlignment = Alignment.Start, + verticalArrangement = Arrangement.Top, + modifier = Modifier + .fillMaxWidth() + .padding(horizontal = 13.dp) + ) { + Box( + modifier = Modifier + .background( + color = Grey100, + shape = RoundedCornerShape(14.dp) + ) + .padding(horizontal = 8.dp, vertical = 5.dp), + contentAlignment = Alignment.Center + ) { + Text( + text = stringResource(id = R.string.dialog_content_color_button), + style = TerningTheme.typography.body5, + color = Grey400, + ) + } + + Box( + modifier = Modifier + .fillMaxWidth() + .padding( + top = 12.dp, + bottom = 23.dp, + ), + contentAlignment = Alignment.Center + ) { + ColorPalette( + initialColor = selectedColor, + onColorSelected = { newColor -> + selectedColor = newColor + } + ) + } + HorizontalDivider( + thickness = 1.dp, + color = Grey200, + modifier = Modifier.padding( + bottom = 8.dp + ) + ) + Column( + modifier = Modifier.padding(bottom = 29.dp), + verticalArrangement = Arrangement.spacedBy( + 5.dp, + Alignment.CenterVertically + ), + horizontalAlignment = Alignment.Start, + ) { + InternInfoRow( + title = stringResource(id = com.terning.feature.R.string.intern_info_d_day), + value = deadline + ) + InternInfoRow( + title = stringResource(id = com.terning.feature.R.string.intern_info_working), + value = workingPeriod + ) + InternInfoRow( + title = stringResource(id = com.terning.feature.R.string.intern_info_start_date), + value = "${startYear}년 ${startMonth}월" + ) + } + } + + if(isScrapped) { + DetailScrapButton( + onClickNavigateButton = onClickNavigateButton, + onClickColorChangeButton = { onClickColorChangeButton(selectedColor) } + ) + } else { + NewScrapButton(onClickScrapButton = onClickScrapButton) + } + } + } +} + +@Composable +private fun NewScrapButton( + onClickScrapButton: () -> Unit, + modifier: Modifier = Modifier +) { + RoundButton( + style = TerningTheme.typography.button3, + paddingVertical = 12.dp, + cornerRadius = 8.dp, + text = R.string.dialog_scrap_button, + onButtonClick = onClickScrapButton, + modifier = modifier + ) +} + +@Composable +private fun DetailScrapButton( + onClickNavigateButton: () -> Unit, + onClickColorChangeButton: () -> Unit, + modifier: Modifier = Modifier +){ + Row ( + modifier = modifier + ){ + RoundButton( + style = TerningTheme.typography.button3, + paddingVertical = 12.dp, + cornerRadius = 8.dp, + text = R.string.dialog_content_calendar_color_change, + onButtonClick = onClickColorChangeButton, + modifier = Modifier.weight(1f) + ) + Spacer(modifier = Modifier.width(8.dp)) + RoundButton( + style = TerningTheme.typography.button3, + paddingVertical = 12.dp, + cornerRadius = 8.dp, + text = R.string.dialog_scrap_move_to_intern, + onButtonClick = onClickNavigateButton, + modifier = Modifier.weight(1f) + ) + } +} + +@Preview(showBackground = true) +@Composable +private fun ScrapDialogPreview() { + TerningPointTheme { + ScrapDialogScreen( + title = "터닝 하반기 채용", + scrapColor = Color.Red, + deadline = "2024/09/07", + startYear = 2024, + startMonth = 11, + workingPeriod = "2개월", + companyImage = "", + isScrapped = false, + onClickNavigateButton = {}, + onClickColorChangeButton = {}, + onClickScrapButton = {} + ) + } +} + +@Preview(showBackground = true) +@Composable +private fun NewScrapButtonPreview() { + TerningPointTheme { + NewScrapButton( + onClickScrapButton = {} + ) + } +} + +@Preview(showBackground = true) +@Composable +private fun DetailScrapButtonPreview() { + TerningPointTheme { + DetailScrapButton( + onClickNavigateButton = {}, + onClickColorChangeButton = {} + ) + } +} \ No newline at end of file diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogSideEffect.kt b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogSideEffect.kt new file mode 100644 index 000000000..e964bf632 --- /dev/null +++ b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogSideEffect.kt @@ -0,0 +1,9 @@ +package com.terning.feature.dialog.detail + +sealed class ScrapDialogSideEffect{ + data class ShowToast(val message: Int): ScrapDialogSideEffect() + data object DismissDialog : ScrapDialogSideEffect() + data object ScrappedAnnouncement : ScrapDialogSideEffect() + data object ChangedColor: ScrapDialogSideEffect() + data object NavigateToDetail : ScrapDialogSideEffect() +} \ No newline at end of file diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogViewModel.kt b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogViewModel.kt new file mode 100644 index 000000000..badacec4c --- /dev/null +++ b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogViewModel.kt @@ -0,0 +1,74 @@ +package com.terning.feature.dialog.detail + +import androidx.compose.ui.graphics.Color +import androidx.lifecycle.ViewModel +import androidx.lifecycle.viewModelScope +import com.terning.core.designsystem.theme.CalBlue1 +import com.terning.core.designsystem.theme.CalBlue2 +import com.terning.core.designsystem.theme.CalGreen1 +import com.terning.core.designsystem.theme.CalGreen2 +import com.terning.core.designsystem.theme.CalOrange1 +import com.terning.core.designsystem.theme.CalOrange2 +import com.terning.core.designsystem.theme.CalPink +import com.terning.core.designsystem.theme.CalPurple +import com.terning.core.designsystem.theme.CalRed +import com.terning.core.designsystem.theme.CalYellow +import com.terning.domain.entity.CalendarScrapRequest +import com.terning.domain.repository.ScrapRepository +import com.terning.feature.R +import dagger.hilt.android.lifecycle.HiltViewModel +import kotlinx.coroutines.flow.MutableSharedFlow +import kotlinx.coroutines.flow.asSharedFlow +import kotlinx.coroutines.launch +import javax.inject.Inject + +@HiltViewModel +class ScrapDialogViewModel @Inject constructor( + private val scrapRepository: ScrapRepository +) : ViewModel() { + private var _sideEffect: MutableSharedFlow = MutableSharedFlow() + val sideEffect = _sideEffect.asSharedFlow() + + fun navigateToDetail() = viewModelScope.launch { + _sideEffect.emit(ScrapDialogSideEffect.NavigateToDetail) + } + + fun postScrap(id: Long, color: Color) { + val colorIndex = getColorIndex(color) + viewModelScope.launch { + scrapRepository.postScrap(CalendarScrapRequest(id, colorIndex)) + .onSuccess { + with(_sideEffect) { + emit(ScrapDialogSideEffect.ShowToast(R.string.intern_scrap_add_toast_message)) + emit(ScrapDialogSideEffect.ScrappedAnnouncement) + } + }.onFailure { + _sideEffect.emit(ScrapDialogSideEffect.ShowToast(R.string.server_failure)) + } + } + } + + fun patchScrap(scrapId: Long, color: Color) = viewModelScope.launch { + val colorIndex = getColorIndex(color) + scrapRepository.patchScrap(CalendarScrapRequest(scrapId, colorIndex)) + .onSuccess { + _sideEffect.emit(ScrapDialogSideEffect.ChangedColor) + }.onFailure { + _sideEffect.emit(ScrapDialogSideEffect.ShowToast(R.string.server_failure)) + } + } + + + private fun getColorIndex(color: Color): Int = listOf( + CalRed, + CalOrange1, + CalOrange2, + CalYellow, + CalGreen1, + CalGreen2, + CalBlue1, + CalBlue2, + CalPurple, + CalPink + ).indexOf(color) +} \ No newline at end of file From 72279a2c06095f7c05bda6e6f2728c73963dd86b Mon Sep 17 00:00:00 2001 From: boiledegg Date: Sun, 1 Sep 2024 17:15:46 +0900 Subject: [PATCH 05/26] =?UTF-8?q?[FEAT/#212]=20=EA=B3=B5=EA=B3=A0=20?= =?UTF-8?q?=EB=8B=A4=EC=9D=B4=EC=96=BC=EB=A1=9C=EA=B7=B8=EC=99=80=20?= =?UTF-8?q?=EB=8B=AC=EB=A0=A5=20=EC=97=B0=EB=8F=99?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- core/src/main/res/values/strings.xml | 6 +-- .../calendar/list/CalendarListScreen.kt | 44 +++++++++++++------ .../calendar/week/CalendarWeekScreen.kt | 37 ++++++++++++++-- 3 files changed, 67 insertions(+), 20 deletions(-) diff --git a/core/src/main/res/values/strings.xml b/core/src/main/res/values/strings.xml index 71380b12d..ad3c916f7 100644 --- a/core/src/main/res/values/strings.xml +++ b/core/src/main/res/values/strings.xml @@ -44,14 +44,14 @@ 공고를 캘린더에 스크랩하시겠어요? - 색상 - 색상 저장하기 + 스크랩 색상 + 색상 변경하기 내 캘린더에 스크랩하기 관심 공고가 캘린더에서 사라져요! 스크랩을 취소하시겠어요? 스크랩 취소하기 내가 스크랩한 관심 공고에요! - 공고 상세 정보 보러가기 + 공고 상세 정보 보기 오늘 마감되는 공고예요! diff --git a/feature/src/main/java/com/terning/feature/calendar/list/CalendarListScreen.kt b/feature/src/main/java/com/terning/feature/calendar/list/CalendarListScreen.kt index e7184ccaa..5cd132dfd 100644 --- a/feature/src/main/java/com/terning/feature/calendar/list/CalendarListScreen.kt +++ b/feature/src/main/java/com/terning/feature/calendar/list/CalendarListScreen.kt @@ -45,6 +45,7 @@ import com.terning.feature.calendar.calendar.model.CalendarModel.Companion.getLo import com.terning.feature.calendar.list.component.CalendarScrapList import com.terning.feature.calendar.list.model.CalendarListUiState import com.terning.feature.dialog.cancel.ScrapCancelDialog +import com.terning.feature.dialog.detail.ScrapDialog import kotlinx.coroutines.flow.distinctUntilChanged import java.time.LocalDate @@ -85,13 +86,18 @@ fun CalendarListRoute( listState = listState, uiState = uiState, modifier = modifier, - navigateToAnnouncement = navigateToAnnouncement, + navigateToAnnouncement = { announcementId -> + navigateToAnnouncement(announcementId) + viewModel.updateInternDialogVisibility(false) + }, onDismissCancelDialog = { isCancelled -> viewModel.updateScrapCancelDialogVisibility(false) if (isCancelled) { viewModel.getScrapMonthList(uiState.currentDate) } }, onDismissInternDialog = { viewModel.updateInternDialogVisibility(false) }, - onClickChangeColor = { newColor -> viewModel.patchScrap(newColor) }, + onClickChangeColor = { + viewModel.getScrapMonthList(uiState.currentDate) + }, onClickScrapButton = { scrapId -> with(viewModel){ updateScrapId(scrapId) @@ -115,7 +121,7 @@ private fun CalendarListScreen( navigateToAnnouncement: (Long) -> Unit, onDismissCancelDialog: (Boolean) -> Unit, onDismissInternDialog: () -> Unit, - onClickChangeColor: (Color) -> Unit, + onClickChangeColor: () -> Unit, onClickInternship: (CalendarScrapDetail) -> Unit, onClickScrapButton: (Long) -> Unit, modifier: Modifier = Modifier @@ -203,16 +209,28 @@ private fun CalendarListScreen( } if (uiState.internDialogVisibility) { - CalendarDetailDialog( - deadline = uiState.currentDate.getFullDateStringInKorean(), - scrapDetailModel = uiState.internshipModel, - onDismissRequest = onDismissInternDialog, - onClickChangeColorButton = onClickChangeColor, - onClickNavigateButton = { announcementId -> - navigateToAnnouncement(announcementId) - onDismissInternDialog() - } - ) + uiState.internshipModel?.let { + val scrapColor = Color( + android.graphics.Color.parseColor( + uiState.internshipModel.color + ) + ) + ScrapDialog( + title = uiState.internshipModel.title, + scrapColor = scrapColor, + deadline = uiState.currentDate.getFullDateStringInKorean(), + startYear = uiState.internshipModel.startYear, + startMonth = uiState.internshipModel.startMonth, + workingPeriod = uiState.internshipModel.workingPeriod, + scrapId = uiState.internshipModel.scrapId, + internshipAnnouncementId = uiState.internshipModel.internshipAnnouncementId, + companyImage = uiState.internshipModel.companyImage, + isScrapped = true, + onDismissRequest = onDismissInternDialog, + onClickChangeColor = onClickChangeColor, + onClickNavigateButton = navigateToAnnouncement + ) + } } } diff --git a/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekScreen.kt b/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekScreen.kt index 2ccb983d4..c5dcbffb1 100644 --- a/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekScreen.kt +++ b/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekScreen.kt @@ -44,6 +44,7 @@ import com.terning.feature.calendar.list.component.CalendarScrapList import com.terning.feature.calendar.week.component.HorizontalCalendarWeek import com.terning.feature.calendar.week.model.CalendarWeekUiState import com.terning.feature.dialog.cancel.ScrapCancelDialog +import com.terning.feature.dialog.detail.ScrapDialog import okhttp3.internal.toImmutableList import java.time.LocalDate @@ -81,7 +82,10 @@ fun CalendarWeekRoute( uiState = uiState, selectedDate = calendarUiState.selectedDate, updateSelectedDate = updateSelectedDate, - navigateToAnnouncement = navigateToAnnouncement, + navigateToAnnouncement = { announcementId -> + navigateToAnnouncement(announcementId) + viewModel.updateInternDialogVisibility(false) + }, onDismissCancelDialog = { isCancelled -> viewModel.updateScrapCancelDialogVisibility(false) if (isCancelled) { @@ -89,7 +93,7 @@ fun CalendarWeekRoute( } }, onDismissInternDialog = { viewModel.updateInternDialogVisibility(false) }, - onClickChangeColor = { viewModel.patchScrap(it) }, + onClickChangeColor = { viewModel.getScrapWeekList(uiState.selectedDate) }, onClickScrapCancel = { uiState.scrapId?.let { viewModel.deleteScrap(it) } }, onClickScrapButton = { scrapId -> with(viewModel) { @@ -113,7 +117,7 @@ private fun CalendarWeekScreen( updateSelectedDate: (LocalDate) -> Unit, onDismissCancelDialog: (Boolean) -> Unit, onDismissInternDialog: () -> Unit, - onClickChangeColor: (Color) -> Unit, + onClickChangeColor: () -> Unit, onClickScrapCancel: () -> Unit, onClickInternship: (CalendarScrapDetail) -> Unit, onClickScrapButton: (Long) -> Unit, @@ -201,6 +205,31 @@ private fun CalendarWeekScreen( } if (uiState.internDialogVisibility) { + uiState.internshipModel?.let { + val scrapColor = Color( + android.graphics.Color.parseColor( + uiState.internshipModel.color + ) + ) + ScrapDialog( + title = uiState.internshipModel.title, + scrapColor = scrapColor, + deadline = uiState.selectedDate.getFullDateStringInKorean(), + startYear = uiState.internshipModel.startYear, + startMonth = uiState.internshipModel.startMonth, + workingPeriod = uiState.internshipModel.workingPeriod, + scrapId = uiState.internshipModel.scrapId, + internshipAnnouncementId = uiState.internshipModel.internshipAnnouncementId, + companyImage = uiState.internshipModel.companyImage, + isScrapped = true, + onDismissRequest = onDismissInternDialog, + onClickChangeColor = onClickChangeColor, + onClickNavigateButton = navigateToAnnouncement + ) + } + } + + /*if (uiState.internDialogVisibility) { CalendarDetailDialog( deadline = uiState.selectedDate.getFullDateStringInKorean(), scrapDetailModel = uiState.internshipModel, @@ -211,7 +240,7 @@ private fun CalendarWeekScreen( onDismissInternDialog() } ) - } + }*/ } @Composable From 610a9a1227666cb9b825aa3f306e29bc28f4e1d9 Mon Sep 17 00:00:00 2001 From: boiledegg Date: Sun, 1 Sep 2024 17:24:39 +0900 Subject: [PATCH 06/26] =?UTF-8?q?[DEL/#212]=20=EC=8A=A4=ED=81=AC=EB=9E=A9?= =?UTF-8?q?=20=EA=B4=80=EB=A6=AC=20=EC=BD=94=EB=93=9C=20=EC=A0=9C=EA=B1=B0?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../component/CalendarCancelDialog.kt | 34 --- .../component/CalendarDetailDialog.kt | 250 ------------------ .../calendar/list/CalendarListScreen.kt | 1 - .../calendar/list/CalendarListViewModel.kt | 52 +--- .../calendar/week/CalendarWeekScreen.kt | 14 - .../calendar/week/CalendarWeekViewModel.kt | 50 +--- 6 files changed, 2 insertions(+), 399 deletions(-) delete mode 100644 feature/src/main/java/com/terning/feature/calendar/calendar/component/CalendarCancelDialog.kt delete mode 100644 feature/src/main/java/com/terning/feature/calendar/calendar/component/CalendarDetailDialog.kt diff --git a/feature/src/main/java/com/terning/feature/calendar/calendar/component/CalendarCancelDialog.kt b/feature/src/main/java/com/terning/feature/calendar/calendar/component/CalendarCancelDialog.kt deleted file mode 100644 index 33b0e762a..000000000 --- a/feature/src/main/java/com/terning/feature/calendar/calendar/component/CalendarCancelDialog.kt +++ /dev/null @@ -1,34 +0,0 @@ -package com.terning.feature.calendar.calendar.component - -import androidx.compose.runtime.Composable -import androidx.compose.ui.tooling.preview.Preview -import com.terning.core.designsystem.component.dialog.ScrapCancelDialogContent -import com.terning.core.designsystem.component.dialog.TerningBasicDialog -import com.terning.core.designsystem.theme.TerningPointTheme - -@Composable -fun CalendarCancelDialog( - onDismissRequest: () -> Unit, - onClickScrapCancel: () -> Unit, -) { - TerningBasicDialog( - onDismissRequest = onDismissRequest - ) { - ScrapCancelDialogContent( - onClickScrapCancel = onClickScrapCancel - ) - } -} - - -@Preview(showBackground = true) -@Composable -fun TerningBasicDialogPreview() { - TerningPointTheme { - TerningBasicDialog( - onDismissRequest = {}, - ) { - ScrapCancelDialogContent() - } - } -} \ No newline at end of file diff --git a/feature/src/main/java/com/terning/feature/calendar/calendar/component/CalendarDetailDialog.kt b/feature/src/main/java/com/terning/feature/calendar/calendar/component/CalendarDetailDialog.kt deleted file mode 100644 index caadbfd19..000000000 --- a/feature/src/main/java/com/terning/feature/calendar/calendar/component/CalendarDetailDialog.kt +++ /dev/null @@ -1,250 +0,0 @@ -package com.terning.feature.calendar.calendar.component - -import androidx.compose.foundation.background -import androidx.compose.foundation.border -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.Spacer -import androidx.compose.foundation.layout.aspectRatio -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.layout.wrapContentSize -import androidx.compose.foundation.shape.RoundedCornerShape -import androidx.compose.material3.HorizontalDivider -import androidx.compose.material3.Icon -import androidx.compose.material3.Text -import androidx.compose.runtime.Composable -import androidx.compose.runtime.getValue -import androidx.compose.runtime.mutableStateOf -import androidx.compose.runtime.remember -import androidx.compose.runtime.setValue -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.res.painterResource -import androidx.compose.ui.res.stringResource -import androidx.compose.ui.text.style.TextAlign -import androidx.compose.ui.unit.dp -import coil.compose.AsyncImage -import coil.request.ImageRequest -import com.terning.core.R -import com.terning.core.designsystem.component.button.RoundButton -import com.terning.core.designsystem.component.dialog.TerningBasicDialog -import com.terning.core.designsystem.component.item.ColorPalette -import com.terning.core.designsystem.theme.Grey200 -import com.terning.core.designsystem.theme.Grey350 -import com.terning.core.designsystem.theme.Grey500 -import com.terning.core.designsystem.theme.TerningMain -import com.terning.core.designsystem.theme.TerningTheme -import com.terning.core.designsystem.theme.White -import com.terning.core.extension.noRippleClickable -import com.terning.domain.entity.CalendarScrapDetail -import com.terning.feature.intern.component.InternInfoRow -import java.time.LocalDate - -@Composable -fun CalendarDetailDialog( - deadline: String, - scrapDetailModel: CalendarScrapDetail?, - onDismissRequest: () -> Unit, - onClickChangeColorButton: (Color) -> Unit, - onClickNavigateButton: (Long) -> Unit, -) { - TerningBasicDialog( - onDismissRequest = onDismissRequest - ) { - InternDialogContent( - deadline = deadline, - scrapDetailModel = scrapDetailModel, - onClickChangeColorButton = onClickChangeColorButton, - onClickNavigateButton = onClickNavigateButton - ) - } -} - - -@Composable -private fun InternDialogContent( - deadline: String, - scrapDetailModel: CalendarScrapDetail?, - onClickChangeColorButton: (Color) -> Unit, - onClickNavigateButton: (Long) -> Unit -) { - var isPaletteOpen by remember { mutableStateOf(false) } - var selectedColor by remember { - mutableStateOf( - Color( - android.graphics.Color.parseColor( - scrapDetailModel?.color - ) - ) - ) - } - - Box( - modifier = Modifier - .wrapContentSize() - .padding(top = 32.dp), - contentAlignment = Alignment.TopCenter - ) { - Column( - modifier = Modifier - .padding(horizontal = 11.dp), - horizontalAlignment = Alignment.CenterHorizontally - ) { - AsyncImage( - model = ImageRequest.Builder(LocalContext.current) - .data(scrapDetailModel?.companyImage) - .build(), - contentDescription = scrapDetailModel?.title, - contentScale = ContentScale.Fit, - modifier = Modifier - .width(80.dp) - .aspectRatio(1f) - .clip(RoundedCornerShape(15.dp)) - .border( - width = 1.dp, - color = TerningMain, - shape = RoundedCornerShape(size = 15.dp) - ) - ) - Text( - text = scrapDetailModel?.title.orEmpty(), - textAlign = TextAlign.Center, - style = TerningTheme.typography.title4, - color = Grey500, - modifier = Modifier.padding(top = 20.dp) - ) - Text( - text = stringResource(id = R.string.dialog_scrap_mine), - style = TerningTheme.typography.body5, - color = Grey350, - modifier = Modifier.padding( - top = 4.dp - ) - ) - Spacer(modifier = Modifier.height(26.dp)) - Column( - horizontalAlignment = Alignment.Start, - verticalArrangement = Arrangement.Top, - modifier = Modifier - .fillMaxWidth() - .padding(horizontal = 13.dp) - ) { - Row( - modifier = Modifier - .background( - color = selectedColor, - shape = RoundedCornerShape(14.dp) - ) - .noRippleClickable { - isPaletteOpen = !isPaletteOpen - }, - verticalAlignment = Alignment.CenterVertically, - horizontalArrangement = Arrangement.Start - ) { - Icon( - painter = painterResource( - id = if (isPaletteOpen) R.drawable.ic_up_22 - else R.drawable.ic_down_22 - ), - contentDescription = stringResource( - id = R.string.dialog_content_color_button - ), - tint = White, - modifier = Modifier.padding( - start = 7.dp, - top = 2.dp, - bottom = 2.dp - ) - ) - Text( - text = stringResource(id = R.string.dialog_content_color_button), - style = TerningTheme.typography.body5, - color = White, - modifier = Modifier.padding(end = 13.dp) - ) - } - HorizontalDivider( - thickness = 1.dp, - color = Grey200, - modifier = Modifier.padding( - top = 11.dp, - bottom = 8.dp - ) - ) - if (isPaletteOpen) { - Box( - modifier = Modifier - .fillMaxWidth() - .padding( - top = 12.dp, - bottom = 23.dp, - ), - contentAlignment = Alignment.Center - ) { - ColorPalette( - initialColor = selectedColor, - onColorSelected = { newColor -> - selectedColor = newColor - } - ) - } - } else { - Text( - text = scrapDetailModel?.dDay.orEmpty(), - style = TerningTheme.typography.body5, - color = TerningMain, - modifier = Modifier.padding(bottom = 9.dp) - ) - Column( - modifier = Modifier.padding(bottom = 29.dp), - verticalArrangement = Arrangement.spacedBy( - 5.dp, - Alignment.CenterVertically - ), - horizontalAlignment = Alignment.Start, - ) { - InternInfoRow( - title = stringResource(id = com.terning.feature.R.string.intern_info_d_day), - value = deadline - ) - InternInfoRow( - title = stringResource(id = com.terning.feature.R.string.intern_info_working), - value = scrapDetailModel?.workingPeriod.orEmpty() - ) - InternInfoRow( - title = stringResource(id = com.terning.feature.R.string.intern_info_start_date), - value = "${scrapDetailModel?.startYear ?: LocalDate.now().year}년 " + - "${scrapDetailModel?.startMonth ?: LocalDate.now().monthValue}월" - ) - } - } - } - RoundButton( - style = TerningTheme.typography.button3, - paddingVertical = 12.dp, - cornerRadius = 8.dp, - text = if (isPaletteOpen) R.string.dialog_content_calendar_color_change - else R.string.dialog_scrap_move_to_intern, - onButtonClick = { - if (isPaletteOpen) { - onClickChangeColorButton(selectedColor) - isPaletteOpen = false - } else { - onClickNavigateButton(scrapDetailModel?.internshipAnnouncementId ?: 0) - } - }, - modifier = Modifier.padding(bottom = 8.dp) - ) - - } - } -} diff --git a/feature/src/main/java/com/terning/feature/calendar/list/CalendarListScreen.kt b/feature/src/main/java/com/terning/feature/calendar/list/CalendarListScreen.kt index 5cd132dfd..91f31fd01 100644 --- a/feature/src/main/java/com/terning/feature/calendar/list/CalendarListScreen.kt +++ b/feature/src/main/java/com/terning/feature/calendar/list/CalendarListScreen.kt @@ -39,7 +39,6 @@ import com.terning.core.extension.toast import com.terning.core.state.UiState import com.terning.domain.entity.CalendarScrapDetail import com.terning.feature.R -import com.terning.feature.calendar.calendar.component.CalendarDetailDialog import com.terning.feature.calendar.calendar.model.CalendarDefaults.flingBehavior import com.terning.feature.calendar.calendar.model.CalendarModel.Companion.getLocalDateByPage import com.terning.feature.calendar.list.component.CalendarScrapList diff --git a/feature/src/main/java/com/terning/feature/calendar/list/CalendarListViewModel.kt b/feature/src/main/java/com/terning/feature/calendar/list/CalendarListViewModel.kt index a178a7f4d..248394035 100644 --- a/feature/src/main/java/com/terning/feature/calendar/list/CalendarListViewModel.kt +++ b/feature/src/main/java/com/terning/feature/calendar/list/CalendarListViewModel.kt @@ -34,8 +34,7 @@ import javax.inject.Inject @HiltViewModel class CalendarListViewModel @Inject constructor( - private val calendarRepository: CalendarRepository, - private val scrapRepository: ScrapRepository + private val calendarRepository: CalendarRepository ): ViewModel(){ private val _uiState = MutableStateFlow(CalendarListUiState()) val uiState = _uiState.asStateFlow() @@ -110,53 +109,4 @@ class CalendarListViewModel @Inject constructor( } ) } - - fun deleteScrap( - scrapId: Long - ) = viewModelScope.launch { - _uiState.value.loadState - .takeIf { it is UiState.Success } - ?.let { CalendarScrapRequest(scrapId, null) }?.let { scrapRequestModel -> - scrapRepository.deleteScrap( - scrapRequestModel - ).onSuccess { - runCatching { - getScrapMonthList(_uiState.value.currentDate) - }.onSuccess { - updateScrapCancelDialogVisibility(false) - } - }.onFailure { - _sideEffect.emit(CalendarListSideEffect.ShowToast(R.string.server_failure)) - } - } - } - - fun patchScrap( - color: Color - ) = viewModelScope.launch { - val scrapId = _uiState.value.internshipModel?.scrapId ?: 0 - val colorIndex = getColorIndex(color) - - scrapRepository.patchScrap(CalendarScrapRequest(scrapId, colorIndex)) - .onSuccess { - runCatching { - getScrapMonthList(_uiState.value.currentDate) - } - }.onFailure { - _sideEffect.emit(CalendarListSideEffect.ShowToast(R.string.server_failure)) - } - } - - private fun getColorIndex(color: Color): Int = listOf( - CalRed, - CalOrange1, - CalOrange2, - CalYellow, - CalGreen1, - CalGreen2, - CalBlue1, - CalBlue2, - CalPurple, - CalPink - ).indexOf(color) } \ No newline at end of file diff --git a/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekScreen.kt b/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekScreen.kt index c5dcbffb1..09c4abcb3 100644 --- a/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekScreen.kt +++ b/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekScreen.kt @@ -38,7 +38,6 @@ import com.terning.core.extension.toast import com.terning.core.state.UiState import com.terning.domain.entity.CalendarScrapDetail import com.terning.feature.R -import com.terning.feature.calendar.calendar.component.CalendarDetailDialog import com.terning.feature.calendar.calendar.model.CalendarUiState import com.terning.feature.calendar.list.component.CalendarScrapList import com.terning.feature.calendar.week.component.HorizontalCalendarWeek @@ -228,19 +227,6 @@ private fun CalendarWeekScreen( ) } } - - /*if (uiState.internDialogVisibility) { - CalendarDetailDialog( - deadline = uiState.selectedDate.getFullDateStringInKorean(), - scrapDetailModel = uiState.internshipModel, - onDismissRequest = onDismissInternDialog, - onClickChangeColorButton = onClickChangeColor, - onClickNavigateButton = { announcementId -> - navigateToAnnouncement(announcementId) - onDismissInternDialog() - } - ) - }*/ } @Composable diff --git a/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekViewModel.kt b/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekViewModel.kt index 07337f7db..3f611a80b 100644 --- a/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekViewModel.kt +++ b/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekViewModel.kt @@ -34,8 +34,7 @@ import javax.inject.Inject @HiltViewModel class CalendarWeekViewModel @Inject constructor( - private val calendarRepository: CalendarRepository, - private val scrapRepository: ScrapRepository + private val calendarRepository: CalendarRepository ): ViewModel() { private val _uiState = MutableStateFlow(CalendarWeekUiState()) val uiState = _uiState.asStateFlow() @@ -105,51 +104,4 @@ class CalendarWeekViewModel @Inject constructor( } ) } - - fun deleteScrap(scrapId: Long) = viewModelScope.launch { - _uiState.value.loadState - .takeIf { it is UiState.Success } - ?.let { CalendarScrapRequest(scrapId, null) }?.let { scrapRequestModel -> - scrapRepository.deleteScrap( - scrapRequestModel - ).onSuccess { - runCatching { - getScrapWeekList(selectedDate = _uiState.value.selectedDate) - }.onSuccess { - updateScrapCancelDialogVisibility(false) - } - }.onFailure { - _sideEffect.emit( - CalendarWeekSideEffect.ShowToast(R.string.server_failure) - ) - } - } - } - - fun patchScrap(color: Color) = viewModelScope.launch { - val scrapId = _uiState.value.internshipModel?.scrapId ?: 0 - val colorIndex = getColorIndex(color) - - scrapRepository.patchScrap(CalendarScrapRequest(scrapId, colorIndex)) - .onSuccess { - runCatching { - getScrapWeekList(selectedDate = _uiState.value.selectedDate) - } - }.onFailure { - _sideEffect.emit(CalendarWeekSideEffect.ShowToast(R.string.server_failure)) - } - } - - private fun getColorIndex(color: Color): Int = listOf( - CalRed, - CalOrange1, - CalOrange2, - CalYellow, - CalGreen1, - CalGreen2, - CalBlue1, - CalBlue2, - CalPurple, - CalPink - ).indexOf(color) } \ No newline at end of file From b483d5623165b63ce7ffa02391b62cc1e2d9b74e Mon Sep 17 00:00:00 2001 From: boiledegg Date: Sun, 1 Sep 2024 17:27:42 +0900 Subject: [PATCH 07/26] =?UTF-8?q?[DEL/#212]=20=EB=9D=BC=EC=9D=B4=EB=B8=8C?= =?UTF-8?q?=EB=9F=AC=EB=A6=AC=20import=20=EC=A0=95=EB=A6=AC?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../feature/calendar/list/CalendarListViewModel.kt | 13 ------------- .../feature/calendar/week/CalendarWeekScreen.kt | 2 -- .../feature/calendar/week/CalendarWeekViewModel.kt | 13 ------------- .../feature/dialog/cancel/ScrapCancelViewModel.kt | 1 - 4 files changed, 29 deletions(-) diff --git a/feature/src/main/java/com/terning/feature/calendar/list/CalendarListViewModel.kt b/feature/src/main/java/com/terning/feature/calendar/list/CalendarListViewModel.kt index 248394035..cfd4f7876 100644 --- a/feature/src/main/java/com/terning/feature/calendar/list/CalendarListViewModel.kt +++ b/feature/src/main/java/com/terning/feature/calendar/list/CalendarListViewModel.kt @@ -1,23 +1,10 @@ package com.terning.feature.calendar.list -import androidx.compose.ui.graphics.Color import androidx.lifecycle.ViewModel import androidx.lifecycle.viewModelScope -import com.terning.core.designsystem.theme.CalBlue1 -import com.terning.core.designsystem.theme.CalBlue2 -import com.terning.core.designsystem.theme.CalGreen1 -import com.terning.core.designsystem.theme.CalGreen2 -import com.terning.core.designsystem.theme.CalOrange1 -import com.terning.core.designsystem.theme.CalOrange2 -import com.terning.core.designsystem.theme.CalPink -import com.terning.core.designsystem.theme.CalPurple -import com.terning.core.designsystem.theme.CalRed -import com.terning.core.designsystem.theme.CalYellow import com.terning.core.state.UiState import com.terning.domain.entity.CalendarScrapDetail -import com.terning.domain.entity.CalendarScrapRequest import com.terning.domain.repository.CalendarRepository -import com.terning.domain.repository.ScrapRepository import com.terning.feature.R import com.terning.feature.calendar.list.model.CalendarListUiState import dagger.hilt.android.lifecycle.HiltViewModel diff --git a/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekScreen.kt b/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekScreen.kt index 09c4abcb3..a994b1916 100644 --- a/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekScreen.kt +++ b/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekScreen.kt @@ -93,7 +93,6 @@ fun CalendarWeekRoute( }, onDismissInternDialog = { viewModel.updateInternDialogVisibility(false) }, onClickChangeColor = { viewModel.getScrapWeekList(uiState.selectedDate) }, - onClickScrapCancel = { uiState.scrapId?.let { viewModel.deleteScrap(it) } }, onClickScrapButton = { scrapId -> with(viewModel) { updateScrapId(scrapId) @@ -117,7 +116,6 @@ private fun CalendarWeekScreen( onDismissCancelDialog: (Boolean) -> Unit, onDismissInternDialog: () -> Unit, onClickChangeColor: () -> Unit, - onClickScrapCancel: () -> Unit, onClickInternship: (CalendarScrapDetail) -> Unit, onClickScrapButton: (Long) -> Unit, navigateToAnnouncement: (Long) -> Unit, diff --git a/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekViewModel.kt b/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekViewModel.kt index 3f611a80b..8374a610b 100644 --- a/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekViewModel.kt +++ b/feature/src/main/java/com/terning/feature/calendar/week/CalendarWeekViewModel.kt @@ -1,23 +1,10 @@ package com.terning.feature.calendar.week -import androidx.compose.ui.graphics.Color import androidx.lifecycle.ViewModel import androidx.lifecycle.viewModelScope -import com.terning.core.designsystem.theme.CalBlue1 -import com.terning.core.designsystem.theme.CalBlue2 -import com.terning.core.designsystem.theme.CalGreen1 -import com.terning.core.designsystem.theme.CalGreen2 -import com.terning.core.designsystem.theme.CalOrange1 -import com.terning.core.designsystem.theme.CalOrange2 -import com.terning.core.designsystem.theme.CalPink -import com.terning.core.designsystem.theme.CalPurple -import com.terning.core.designsystem.theme.CalRed -import com.terning.core.designsystem.theme.CalYellow import com.terning.core.state.UiState import com.terning.domain.entity.CalendarScrapDetail -import com.terning.domain.entity.CalendarScrapRequest import com.terning.domain.repository.CalendarRepository -import com.terning.domain.repository.ScrapRepository import com.terning.feature.R import com.terning.feature.calendar.week.model.CalendarWeekUiState import dagger.hilt.android.lifecycle.HiltViewModel diff --git a/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelViewModel.kt b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelViewModel.kt index 5d3c6db22..beb35c20d 100644 --- a/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelViewModel.kt +++ b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelViewModel.kt @@ -5,7 +5,6 @@ import androidx.lifecycle.viewModelScope import com.terning.domain.entity.CalendarScrapRequest import com.terning.domain.repository.ScrapRepository import com.terning.feature.R -import com.terning.feature.calendar.list.CalendarListSideEffect import dagger.hilt.android.lifecycle.HiltViewModel import kotlinx.coroutines.flow.MutableSharedFlow import kotlinx.coroutines.flow.asSharedFlow From 0a9456513f7d39055bb137aaff2bf85b1091b4e9 Mon Sep 17 00:00:00 2001 From: boiledegg Date: Sun, 1 Sep 2024 17:30:11 +0900 Subject: [PATCH 08/26] =?UTF-8?q?[CHORE/#212]=20Resource=20=EC=9D=98?= =?UTF-8?q?=EC=A1=B4=EC=84=B1=20=EC=88=98=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../feature/dialog/cancel/ScrapCancelDialog.kt | 2 +- .../com/terning/feature/dialog/detail/ScrapDialog.kt | 2 +- feature/src/main/res/values/strings.xml | 12 ++++++++++++ 3 files changed, 14 insertions(+), 2 deletions(-) diff --git a/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt index 328656e75..13329120b 100644 --- a/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt +++ b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt @@ -17,7 +17,6 @@ import androidx.compose.ui.unit.dp import androidx.hilt.navigation.compose.hiltViewModel import androidx.lifecycle.compose.LocalLifecycleOwner import androidx.lifecycle.flowWithLifecycle -import com.terning.core.R import com.terning.core.designsystem.component.button.RoundButton import com.terning.core.designsystem.component.dialog.TerningBasicDialog import com.terning.core.designsystem.component.item.TerningLottieAnimation @@ -25,6 +24,7 @@ import com.terning.core.designsystem.theme.Grey350 import com.terning.core.designsystem.theme.Grey500 import com.terning.core.designsystem.theme.TerningPointTheme import com.terning.core.designsystem.theme.TerningTheme +import com.terning.feature.R @Composable fun ScrapCancelDialog( diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt index 30ab99189..76d100325 100644 --- a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt +++ b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt @@ -37,7 +37,6 @@ import androidx.lifecycle.compose.LocalLifecycleOwner import androidx.lifecycle.flowWithLifecycle import coil.compose.AsyncImage import coil.request.ImageRequest -import com.terning.core.R import com.terning.core.designsystem.component.button.RoundButton import com.terning.core.designsystem.component.dialog.TerningBasicDialog import com.terning.core.designsystem.component.item.ColorPalette @@ -49,6 +48,7 @@ import com.terning.core.designsystem.theme.Grey500 import com.terning.core.designsystem.theme.TerningMain import com.terning.core.designsystem.theme.TerningPointTheme import com.terning.core.designsystem.theme.TerningTheme +import com.terning.feature.R import com.terning.feature.intern.component.InternInfoRow diff --git a/feature/src/main/res/values/strings.xml b/feature/src/main/res/values/strings.xml index 187a825d0..0190c1a92 100644 --- a/feature/src/main/res/values/strings.xml +++ b/feature/src/main/res/values/strings.xml @@ -58,6 +58,18 @@ 선택하신 날짜에 지원 마감인 스크랩 공고가 없어요 +%s + + 공고를 캘린더에 스크랩하시겠어요? + 스크랩 색상 + 색상 변경하기 + 내 캘린더에 스크랩하기 + 관심 공고가 캘린더에서 사라져요! + 스크랩을 취소하시겠어요? + 스크랩 취소하기 + 내가 스크랩한 관심 공고에요! + 공고 상세 정보 보기 + 오늘 마감되는 공고예요! + 오늘 마감되는 %s님의 관심 공고 아직 스크랩된 인턴 공고가 없어요!\n관심 공고를 스크랩하면 마감 당일에 알려드릴게요 From 8bb8578e80dfa1f5e374303d673750d8e8bcd880 Mon Sep 17 00:00:00 2001 From: boiledegg Date: Tue, 3 Sep 2024 15:06:35 +0900 Subject: [PATCH 09/26] =?UTF-8?q?[ADD/#212]=20=EC=8A=A4=ED=81=AC=EB=9E=A9?= =?UTF-8?q?=20=EC=83=89=EC=83=81=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../java/com/terning/core/designsystem/theme/Color.kt | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/core/src/main/java/com/terning/core/designsystem/theme/Color.kt b/core/src/main/java/com/terning/core/designsystem/theme/Color.kt index bea5eb805..5376fd788 100644 --- a/core/src/main/java/com/terning/core/designsystem/theme/Color.kt +++ b/core/src/main/java/com/terning/core/designsystem/theme/Color.kt @@ -41,6 +41,14 @@ val CalBlue2 = Color(0xFF4AA9F2) val CalPurple = Color(0xFF9B64E2) val CalPink = Color(0xFFF260AC) +// Calendar Border Color +val CalRedLi = Color(0x7FED4E54) +val CalOrangeLi = Color(0x7FF3A649) +val CalGreenLi = Color(0x7F84D558) +val CalBlueLi = Color(0x7F4AA9F2) +val CalPurpleLi = Color(0x7F9B64E2) +val CalPinkLi = Color(0x7FF260AC) + // Other val WarningRed = Color(0xFFF54645) val SundayRed = Color(0xFFEB1211) From 4dfff2a0f8ba0259e36271f9e8ec87ec19d91452 Mon Sep 17 00:00:00 2001 From: boiledegg Date: Tue, 3 Sep 2024 15:11:39 +0900 Subject: [PATCH 10/26] =?UTF-8?q?[ADD/#212]=20=EC=BB=AC=EB=9F=AC=20?= =?UTF-8?q?=ED=83=80=EC=9E=85=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../java/com/terning/core/type/ColorType.kt | 26 +++++++++++++++++++ 1 file changed, 26 insertions(+) create mode 100644 core/src/main/java/com/terning/core/type/ColorType.kt diff --git a/core/src/main/java/com/terning/core/type/ColorType.kt b/core/src/main/java/com/terning/core/type/ColorType.kt new file mode 100644 index 000000000..bcdd80fb4 --- /dev/null +++ b/core/src/main/java/com/terning/core/type/ColorType.kt @@ -0,0 +1,26 @@ +package com.terning.core.type + +import androidx.compose.ui.graphics.Color +import com.terning.core.designsystem.theme.CalBlue1 +import com.terning.core.designsystem.theme.CalBlue2 +import com.terning.core.designsystem.theme.CalBlueLi +import com.terning.core.designsystem.theme.CalGreen2 +import com.terning.core.designsystem.theme.CalOrange2 +import com.terning.core.designsystem.theme.CalPink +import com.terning.core.designsystem.theme.CalPinkLi +import com.terning.core.designsystem.theme.CalPurple +import com.terning.core.designsystem.theme.CalPurpleLi +import com.terning.core.designsystem.theme.CalRed +import com.terning.core.designsystem.theme.CalRedLi + +enum class ColorType( + main: Color, + sub: Color +) { + RED(main = CalRed, sub = CalRedLi), + ORANGE(main = CalOrange2, sub = CalRedLi), + GREEN(main = CalGreen2, sub = CalRedLi), + BLUE(main = CalBlue2, sub = CalBlueLi), + PURPLE(main = CalPurple, sub = CalPurpleLi), + PINK(main = CalPink, sub = CalPinkLi) +} \ No newline at end of file From 67dfbf9dbb122306ccf46f00fcc65e8c8cf44dec Mon Sep 17 00:00:00 2001 From: boiledegg Date: Tue, 3 Sep 2024 16:10:34 +0900 Subject: [PATCH 11/26] =?UTF-8?q?[ADD/#212]=20=EC=BB=AC=EB=9F=AC=20?= =?UTF-8?q?=EC=BD=94=EB=93=9C=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../terning/core/designsystem/theme/Color.kt | 22 +++++++++++++------ 1 file changed, 15 insertions(+), 7 deletions(-) diff --git a/core/src/main/java/com/terning/core/designsystem/theme/Color.kt b/core/src/main/java/com/terning/core/designsystem/theme/Color.kt index 5376fd788..1ea24c8b6 100644 --- a/core/src/main/java/com/terning/core/designsystem/theme/Color.kt +++ b/core/src/main/java/com/terning/core/designsystem/theme/Color.kt @@ -41,13 +41,21 @@ val CalBlue2 = Color(0xFF4AA9F2) val CalPurple = Color(0xFF9B64E2) val CalPink = Color(0xFFF260AC) -// Calendar Border Color -val CalRedLi = Color(0x7FED4E54) -val CalOrangeLi = Color(0x7FF3A649) -val CalGreenLi = Color(0x7F84D558) -val CalBlueLi = Color(0x7F4AA9F2) -val CalPurpleLi = Color(0x7F9B64E2) -val CalPinkLi = Color(0x7FF260AC) +// Calendar Color (Border) +val CalRedLi = Color(0xFFDD2F36) +val CalOrangeLi = Color(0xFFF39D35) +val CalGreenLi = Color(0xFF74CE44) +val CalBlueLi = Color(0xFF3B9CE8) +val CalPurpleLi = Color(0xFF8D4EDE) +val CalPinkLi = Color(0xFFF9439A) + +// Calendar Color (Background) +val CalRedBc = Color(0x7FED4E54) +val CalOrangeBc = Color(0x7FF3A649) +val CalGreenBc = Color(0x7F84D558) +val CalBlueBc = Color(0x7F4AA9F2) +val CalPurpleBc = Color(0x7F9B64E2) +val CalPinkBc = Color(0x7FF260AC) // Other val WarningRed = Color(0xFFF54645) From 37c8b2b9d629b53f26115f94e957a32cc52ca42e Mon Sep 17 00:00:00 2001 From: boiledegg Date: Tue, 3 Sep 2024 16:10:51 +0900 Subject: [PATCH 12/26] =?UTF-8?q?[FIX/#212]=20=EC=BB=AC=EB=9F=AC=20?= =?UTF-8?q?=ED=83=80=EC=9E=85=20=EC=88=98=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../java/com/terning/core/type/ColorType.kt | 30 +++++++++++++------ 1 file changed, 21 insertions(+), 9 deletions(-) diff --git a/core/src/main/java/com/terning/core/type/ColorType.kt b/core/src/main/java/com/terning/core/type/ColorType.kt index bcdd80fb4..7d35c8f65 100644 --- a/core/src/main/java/com/terning/core/type/ColorType.kt +++ b/core/src/main/java/com/terning/core/type/ColorType.kt @@ -1,26 +1,38 @@ package com.terning.core.type import androidx.compose.ui.graphics.Color -import com.terning.core.designsystem.theme.CalBlue1 import com.terning.core.designsystem.theme.CalBlue2 +import com.terning.core.designsystem.theme.CalBlueBc import com.terning.core.designsystem.theme.CalBlueLi import com.terning.core.designsystem.theme.CalGreen2 +import com.terning.core.designsystem.theme.CalGreenBc +import com.terning.core.designsystem.theme.CalGreenLi import com.terning.core.designsystem.theme.CalOrange2 +import com.terning.core.designsystem.theme.CalOrangeBc +import com.terning.core.designsystem.theme.CalOrangeLi import com.terning.core.designsystem.theme.CalPink +import com.terning.core.designsystem.theme.CalPinkBc import com.terning.core.designsystem.theme.CalPinkLi import com.terning.core.designsystem.theme.CalPurple +import com.terning.core.designsystem.theme.CalPurpleBc import com.terning.core.designsystem.theme.CalPurpleLi import com.terning.core.designsystem.theme.CalRed +import com.terning.core.designsystem.theme.CalRedBc import com.terning.core.designsystem.theme.CalRedLi enum class ColorType( - main: Color, - sub: Color + val main: Color, + val border: Color, + val sub: Color ) { - RED(main = CalRed, sub = CalRedLi), - ORANGE(main = CalOrange2, sub = CalRedLi), - GREEN(main = CalGreen2, sub = CalRedLi), - BLUE(main = CalBlue2, sub = CalBlueLi), - PURPLE(main = CalPurple, sub = CalPurpleLi), - PINK(main = CalPink, sub = CalPinkLi) + RED(main = CalRed, border = CalRedLi, sub = CalRedBc), + ORANGE(main = CalOrange2, border = CalOrangeLi, sub = CalOrangeBc), + GREEN(main = CalGreen2, border = CalGreenLi, sub = CalGreenBc), + BLUE(main = CalBlue2, border = CalBlueLi, sub = CalBlueBc), + PURPLE(main = CalPurple, border = CalPurpleLi, sub = CalPurpleBc), + PINK(main = CalPink, border = CalPinkLi, sub = CalPinkBc); + + companion object { + fun findColorType(mainColor: Color): ColorType? = entries.find { it.main == mainColor } + } } \ No newline at end of file From a7ccfe7d7256202ae6323c0bec7c27526491b2fc Mon Sep 17 00:00:00 2001 From: boiledegg Date: Tue, 3 Sep 2024 16:16:51 +0900 Subject: [PATCH 13/26] =?UTF-8?q?[FEAT/#212]=20=EC=BB=AC=EB=9F=AC=20?= =?UTF-8?q?=ED=8C=94=EB=A0=88=ED=8A=B8=20=EC=88=98=EC=A0=95=20=EB=B0=8F=20?= =?UTF-8?q?=EC=97=B0=EB=8F=99?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../feature/dialog/detail/ScrapDialog.kt | 10 +- .../dialog/detail/component/ColorPalette.kt | 117 ++++++++++++++++++ 2 files changed, 124 insertions(+), 3 deletions(-) create mode 100644 feature/src/main/java/com/terning/feature/dialog/detail/component/ColorPalette.kt diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt index 76d100325..7a8d6dac9 100644 --- a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt +++ b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt @@ -39,7 +39,7 @@ import coil.compose.AsyncImage import coil.request.ImageRequest import com.terning.core.designsystem.component.button.RoundButton import com.terning.core.designsystem.component.dialog.TerningBasicDialog -import com.terning.core.designsystem.component.item.ColorPalette +import com.terning.core.designsystem.theme.CalRed import com.terning.core.designsystem.theme.Grey100 import com.terning.core.designsystem.theme.Grey200 import com.terning.core.designsystem.theme.Grey350 @@ -48,7 +48,9 @@ import com.terning.core.designsystem.theme.Grey500 import com.terning.core.designsystem.theme.TerningMain import com.terning.core.designsystem.theme.TerningPointTheme import com.terning.core.designsystem.theme.TerningTheme +import com.terning.core.type.ColorType import com.terning.feature.R +import com.terning.feature.dialog.detail.component.ColorPalette import com.terning.feature.intern.component.InternInfoRow @@ -123,6 +125,7 @@ private fun ScrapDialogScreen( onClickScrapButton: () -> Unit ) { var selectedColor by remember { mutableStateOf(scrapColor) } + var selectedColorType by remember { mutableStateOf(ColorType.findColorType(selectedColor)?:ColorType.RED) } Box( modifier = Modifier @@ -200,9 +203,10 @@ private fun ScrapDialogScreen( contentAlignment = Alignment.Center ) { ColorPalette( - initialColor = selectedColor, + initialColor = selectedColorType, onColorSelected = { newColor -> - selectedColor = newColor + selectedColor = newColor.main + selectedColorType = newColor } ) } diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/component/ColorPalette.kt b/feature/src/main/java/com/terning/feature/dialog/detail/component/ColorPalette.kt new file mode 100644 index 000000000..4f6df28ee --- /dev/null +++ b/feature/src/main/java/com/terning/feature/dialog/detail/component/ColorPalette.kt @@ -0,0 +1,117 @@ +package com.terning.feature.dialog.detail.component + +import androidx.compose.foundation.Image +import androidx.compose.foundation.background +import androidx.compose.foundation.border +import androidx.compose.foundation.layout.Box +import androidx.compose.foundation.layout.padding +import androidx.compose.foundation.layout.size +import androidx.compose.foundation.layout.wrapContentSize +import androidx.compose.foundation.shape.CircleShape +import androidx.compose.runtime.Composable +import androidx.compose.runtime.getValue +import androidx.compose.runtime.mutableStateOf +import androidx.compose.runtime.remember +import androidx.compose.runtime.setValue +import androidx.compose.ui.Alignment +import androidx.compose.ui.Modifier +import androidx.compose.ui.graphics.Color +import androidx.compose.ui.res.painterResource +import androidx.compose.ui.tooling.preview.Preview +import androidx.compose.ui.unit.dp +import com.terning.core.R +import com.terning.core.designsystem.component.item.RadioButtonGroups +import com.terning.core.extension.noRippleClickable +import com.terning.core.type.ColorType + +@Composable +fun ColorPalette( + initialColor: ColorType, + onColorSelected: (ColorType) -> Unit, + modifier: Modifier = Modifier +) { + var selectedColor by remember { mutableStateOf(initialColor) } + + RadioButtonGroups( + options = ColorType.entries, + gridCellCount = ColorType.entries.size, + onOptionSelected = { color -> + selectedColor = color + onColorSelected(color) + }, + buttonComposable = { colorType, isSelected, onOptionSelected -> + + + ColorButton( + colorType = colorType, + isSelected = isSelected,//selectedColor == colorType, + onColorSelected = onOptionSelected + ) + }, + verticalArrangementSpace = 6.dp, + horizontalArrangementSpace = 0.dp, + modifier = modifier.wrapContentSize() + ) +} + +@Composable +internal fun ColorButton( + colorType: ColorType, + isSelected: Boolean, + onColorSelected: (ColorType) -> Unit, + modifier: Modifier = Modifier, +) { + val externalColor = if (isSelected) colorType.sub else Color.Transparent + val externalBoxModifier = Modifier + .size(38.dp) + .background( + shape = CircleShape, + color = externalColor + ) + val internalBoxModifier = Modifier + .padding(4.dp) + .then( + if (isSelected) + Modifier.border( + width = 2.dp, + color = colorType.border, + shape = CircleShape, + ) + else Modifier + ) + .background( + shape = CircleShape, + color = colorType.main + ) + + + Box( + modifier = modifier.wrapContentSize(), + ) { + Box( + modifier = Modifier + .noRippleClickable { + onColorSelected(colorType) + } + .then(externalBoxModifier) + .then(internalBoxModifier), + contentAlignment = Alignment.Center + ) { + if (isSelected) { + Image( + painter = painterResource(id = R.drawable.ic_color_check), + contentDescription = "" + ) + } + } + } +} + +@Preview(showBackground = true) +@Composable +fun ColorPalettePreview() { + ColorPalette( + initialColor = ColorType.RED, + onColorSelected = {} + ) +} From f856344c10e54ba17abf0d16f7fdb291dbc1b2be Mon Sep 17 00:00:00 2001 From: boiledegg Date: Tue, 3 Sep 2024 22:44:29 +0900 Subject: [PATCH 14/26] =?UTF-8?q?[FEAT/#212]=20=EC=8A=A4=ED=81=AC=EB=9E=A9?= =?UTF-8?q?=20=EB=8B=A4=EC=9D=B4=EC=96=BC=EB=A1=9C=EA=B7=B8=20UI=20?= =?UTF-8?q?=EB=B0=8F=20=EA=B8=B0=EB=8A=A5=20=EC=88=98=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../java/com/terning/core/type/ColorType.kt | 1 + .../dialog/cancel/ScrapCancelDialog.kt | 8 +- .../feature/dialog/detail/ScrapDialog.kt | 99 +++++++++++++------ .../dialog/detail/ScrapDialogViewModel.kt | 45 +++++++-- .../dialog/detail/component/ColorPalette.kt | 15 +-- .../component/ScrapColorChangeButton.kt | 89 +++++++++++++++++ .../dialog/detail/state/ScrapDialogUiState.kt | 10 ++ feature/src/main/res/values/strings.xml | 1 + 8 files changed, 218 insertions(+), 50 deletions(-) create mode 100644 feature/src/main/java/com/terning/feature/dialog/detail/component/ScrapColorChangeButton.kt create mode 100644 feature/src/main/java/com/terning/feature/dialog/detail/state/ScrapDialogUiState.kt diff --git a/core/src/main/java/com/terning/core/type/ColorType.kt b/core/src/main/java/com/terning/core/type/ColorType.kt index 7d35c8f65..073fc15d5 100644 --- a/core/src/main/java/com/terning/core/type/ColorType.kt +++ b/core/src/main/java/com/terning/core/type/ColorType.kt @@ -1,5 +1,6 @@ package com.terning.core.type +import android.util.Log import androidx.compose.ui.graphics.Color import com.terning.core.designsystem.theme.CalBlue2 import com.terning.core.designsystem.theme.CalBlueBc diff --git a/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt index 13329120b..8fb590437 100644 --- a/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt +++ b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt @@ -2,8 +2,10 @@ package com.terning.feature.dialog.cancel import androidx.compose.foundation.layout.Box import androidx.compose.foundation.layout.Column +import androidx.compose.foundation.layout.Spacer import androidx.compose.foundation.layout.aspectRatio import androidx.compose.foundation.layout.fillMaxWidth +import androidx.compose.foundation.layout.height import androidx.compose.foundation.layout.padding import androidx.compose.material3.Text import androidx.compose.runtime.Composable @@ -93,14 +95,16 @@ private fun ScrapCancelScreen( top = 5.dp ) ) + Spacer(modifier = Modifier.height(40.dp)) + RoundButton( style = TerningTheme.typography.button3, paddingVertical = 12.dp, cornerRadius = 8.dp, text = R.string.dialog_scrap_cancel_button, - onButtonClick = onClickScrapCancel, - modifier = Modifier.padding(bottom = 8.dp, top = 40.dp) + onButtonClick = onClickScrapCancel ) + Spacer(modifier = Modifier.height(16.dp)) } } } diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt index 7a8d6dac9..24110f0cb 100644 --- a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt +++ b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt @@ -19,9 +19,6 @@ import androidx.compose.material3.Text import androidx.compose.runtime.Composable import androidx.compose.runtime.LaunchedEffect import androidx.compose.runtime.getValue -import androidx.compose.runtime.mutableStateOf -import androidx.compose.runtime.remember -import androidx.compose.runtime.setValue import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.draw.clip @@ -34,12 +31,12 @@ import androidx.compose.ui.tooling.preview.Preview import androidx.compose.ui.unit.dp import androidx.hilt.navigation.compose.hiltViewModel import androidx.lifecycle.compose.LocalLifecycleOwner +import androidx.lifecycle.compose.collectAsStateWithLifecycle import androidx.lifecycle.flowWithLifecycle import coil.compose.AsyncImage import coil.request.ImageRequest import com.terning.core.designsystem.component.button.RoundButton import com.terning.core.designsystem.component.dialog.TerningBasicDialog -import com.terning.core.designsystem.theme.CalRed import com.terning.core.designsystem.theme.Grey100 import com.terning.core.designsystem.theme.Grey200 import com.terning.core.designsystem.theme.Grey350 @@ -51,7 +48,9 @@ import com.terning.core.designsystem.theme.TerningTheme import com.terning.core.type.ColorType import com.terning.feature.R import com.terning.feature.dialog.detail.component.ColorPalette +import com.terning.feature.dialog.detail.component.ScrapColorChangeButton import com.terning.feature.intern.component.InternInfoRow +import timber.log.Timber @Composable @@ -72,6 +71,7 @@ fun ScrapDialog( viewModel: ScrapDialogViewModel = hiltViewModel() ) { val lifecycleOwner = LocalLifecycleOwner.current + val uiState by viewModel.uiState.collectAsStateWithLifecycle(lifecycleOwner = lifecycleOwner) LaunchedEffect(viewModel.sideEffect, lifecycleOwner) { viewModel.sideEffect.flowWithLifecycle(lifecycle = lifecycleOwner.lifecycle) .collect { sideEffect -> @@ -82,29 +82,57 @@ fun ScrapDialog( is ScrapDialogSideEffect.NavigateToDetail -> onClickNavigateButton( internshipAnnouncementId ) + is ScrapDialogSideEffect.ScrappedAnnouncement -> {} } } } + LaunchedEffect(true) { + with(viewModel) { + val colorType = ColorType.findColorType(scrapColor).takeIf { it != null } + colorType?.let { + updateInitialColorType(colorType) + updateSelectedColorType(colorType) + } + } + } + TerningBasicDialog( onDismissRequest = onDismissRequest ) { ScrapDialogScreen( title = title, - scrapColor = scrapColor, deadline = deadline, startYear = startYear, startMonth = startMonth, workingPeriod = workingPeriod, isScrapped = isScrapped, companyImage = companyImage, - onClickColorChangeButton = { newColor -> + selectedColorType = uiState.selectedColorType, + isColorChanged = uiState.isColorChanged, + isColorChangedOnce = uiState.isColorChangedOnce, + onClickColorButton = { selectedColorType -> + with(viewModel) { + updateSelectedColorType(selectedColorType) + updateIsColorChanged() + } + }, + onClickColorChangeButton = { + with(viewModel) { + patchScrap(scrapId = scrapId, color = uiState.selectedColorType) + setIsColorChangedOnce() + updateInitialColorType(colorType = uiState.selectedColorType) + } onClickChangeColor() - viewModel.patchScrap(scrapId = scrapId, color = newColor) }, onClickNavigateButton = viewModel::navigateToDetail, - onClickScrapButton = { viewModel.postScrap(internshipAnnouncementId, scrapColor) } + onClickScrapButton = { + viewModel.postScrap( + internshipAnnouncementId, + uiState.selectedColorType + ) + } ) } } @@ -113,20 +141,20 @@ fun ScrapDialog( @Composable private fun ScrapDialogScreen( title: String, - scrapColor: Color, deadline: String, startYear: Int, startMonth: Int, workingPeriod: String, isScrapped: Boolean, companyImage: String, + selectedColorType: ColorType, + isColorChanged: Boolean, + isColorChangedOnce: Boolean, + onClickColorButton: (ColorType) -> Unit, onClickNavigateButton: () -> Unit, - onClickColorChangeButton: (Color) -> Unit, + onClickColorChangeButton: () -> Unit, onClickScrapButton: () -> Unit ) { - var selectedColor by remember { mutableStateOf(scrapColor) } - var selectedColorType by remember { mutableStateOf(ColorType.findColorType(selectedColor)?:ColorType.RED) } - Box( modifier = Modifier .wrapContentSize() @@ -203,11 +231,8 @@ private fun ScrapDialogScreen( contentAlignment = Alignment.Center ) { ColorPalette( - initialColor = selectedColorType, - onColorSelected = { newColor -> - selectedColor = newColor.main - selectedColorType = newColor - } + selectedColor = selectedColorType, + onColorSelected = onClickColorButton ) } HorizontalDivider( @@ -226,24 +251,27 @@ private fun ScrapDialogScreen( horizontalAlignment = Alignment.Start, ) { InternInfoRow( - title = stringResource(id = com.terning.feature.R.string.intern_info_d_day), + title = stringResource(id = R.string.intern_info_d_day), value = deadline ) InternInfoRow( - title = stringResource(id = com.terning.feature.R.string.intern_info_working), + title = stringResource(id = R.string.intern_info_working), value = workingPeriod ) InternInfoRow( - title = stringResource(id = com.terning.feature.R.string.intern_info_start_date), + title = stringResource(id = R.string.intern_info_start_date), value = "${startYear}년 ${startMonth}월" ) } } - if(isScrapped) { + if (isScrapped) { DetailScrapButton( + isColorChanged = isColorChanged, + isColorChangedOnce = isColorChangedOnce, onClickNavigateButton = onClickNavigateButton, - onClickColorChangeButton = { onClickColorChangeButton(selectedColor) } + onClickColorChangeButton = onClickColorChangeButton + ) } else { NewScrapButton(onClickScrapButton = onClickScrapButton) @@ -269,18 +297,20 @@ private fun NewScrapButton( @Composable private fun DetailScrapButton( + isColorChanged: Boolean, + isColorChangedOnce: Boolean, onClickNavigateButton: () -> Unit, onClickColorChangeButton: () -> Unit, modifier: Modifier = Modifier -){ - Row ( +) { + Row( modifier = modifier - ){ - RoundButton( - style = TerningTheme.typography.button3, + ) { + ScrapColorChangeButton( + isColorChanged = isColorChanged, + isColorChangedOnce = isColorChangedOnce, paddingVertical = 12.dp, cornerRadius = 8.dp, - text = R.string.dialog_content_calendar_color_change, onButtonClick = onClickColorChangeButton, modifier = Modifier.weight(1f) ) @@ -302,7 +332,6 @@ private fun ScrapDialogPreview() { TerningPointTheme { ScrapDialogScreen( title = "터닝 하반기 채용", - scrapColor = Color.Red, deadline = "2024/09/07", startYear = 2024, startMonth = 11, @@ -311,7 +340,11 @@ private fun ScrapDialogPreview() { isScrapped = false, onClickNavigateButton = {}, onClickColorChangeButton = {}, - onClickScrapButton = {} + onClickScrapButton = {}, + onClickColorButton = {}, + isColorChanged = false, + isColorChangedOnce = true, + selectedColorType = ColorType.RED ) } } @@ -332,7 +365,9 @@ private fun DetailScrapButtonPreview() { TerningPointTheme { DetailScrapButton( onClickNavigateButton = {}, - onClickColorChangeButton = {} + onClickColorChangeButton = {}, + isColorChanged = false, + isColorChangedOnce = false ) } } \ No newline at end of file diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogViewModel.kt b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogViewModel.kt index badacec4c..e0510a3f6 100644 --- a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogViewModel.kt +++ b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogViewModel.kt @@ -13,28 +13,51 @@ import com.terning.core.designsystem.theme.CalPink import com.terning.core.designsystem.theme.CalPurple import com.terning.core.designsystem.theme.CalRed import com.terning.core.designsystem.theme.CalYellow +import com.terning.core.type.ColorType import com.terning.domain.entity.CalendarScrapRequest import com.terning.domain.repository.ScrapRepository import com.terning.feature.R +import com.terning.feature.dialog.detail.state.ScrapDialogUiState import dagger.hilt.android.lifecycle.HiltViewModel import kotlinx.coroutines.flow.MutableSharedFlow +import kotlinx.coroutines.flow.MutableStateFlow import kotlinx.coroutines.flow.asSharedFlow +import kotlinx.coroutines.flow.asStateFlow +import kotlinx.coroutines.flow.update import kotlinx.coroutines.launch +import timber.log.Timber import javax.inject.Inject @HiltViewModel class ScrapDialogViewModel @Inject constructor( private val scrapRepository: ScrapRepository ) : ViewModel() { + private var _uiState = MutableStateFlow(ScrapDialogUiState()) + val uiState = _uiState.asStateFlow() + private var _sideEffect: MutableSharedFlow = MutableSharedFlow() val sideEffect = _sideEffect.asSharedFlow() - fun navigateToDetail() = viewModelScope.launch { - _sideEffect.emit(ScrapDialogSideEffect.NavigateToDetail) + fun updateInitialColorType(colorType: ColorType) { + _uiState.update { currentState -> + currentState.copy(initialColorType = colorType) + } + } + + fun updateSelectedColorType(colorType: ColorType) { + _uiState.update { currentState -> + if (currentState.selectedColorType == colorType) return@update currentState + currentState.copy(selectedColorType = colorType) + } + } + + fun updateIsColorChanged() = _uiState.update { currentState -> + val isColorChanged = _uiState.value.selectedColorType != _uiState.value.initialColorType + currentState.copy(isColorChanged = isColorChanged) } - fun postScrap(id: Long, color: Color) { - val colorIndex = getColorIndex(color) + fun postScrap(id: Long, color: ColorType) { + val colorIndex = getColorIndex(color.main) viewModelScope.launch { scrapRepository.postScrap(CalendarScrapRequest(id, colorIndex)) .onSuccess { @@ -48,16 +71,26 @@ class ScrapDialogViewModel @Inject constructor( } } - fun patchScrap(scrapId: Long, color: Color) = viewModelScope.launch { - val colorIndex = getColorIndex(color) + fun patchScrap(scrapId: Long, color: ColorType) = viewModelScope.launch { + val colorIndex = getColorIndex(color.main) scrapRepository.patchScrap(CalendarScrapRequest(scrapId, colorIndex)) .onSuccess { _sideEffect.emit(ScrapDialogSideEffect.ChangedColor) + }.onFailure { _sideEffect.emit(ScrapDialogSideEffect.ShowToast(R.string.server_failure)) } } + fun setIsColorChangedOnce() = _uiState.update { currentState -> + if (!currentState.isColorChangedOnce) return@update currentState + currentState.copy(isColorChangedOnce = true) + } + + fun navigateToDetail() = viewModelScope.launch { + _sideEffect.emit(ScrapDialogSideEffect.NavigateToDetail) + } + private fun getColorIndex(color: Color): Int = listOf( CalRed, diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/component/ColorPalette.kt b/feature/src/main/java/com/terning/feature/dialog/detail/component/ColorPalette.kt index 4f6df28ee..13cbdfa0f 100644 --- a/feature/src/main/java/com/terning/feature/dialog/detail/component/ColorPalette.kt +++ b/feature/src/main/java/com/terning/feature/dialog/detail/component/ColorPalette.kt @@ -23,28 +23,23 @@ import com.terning.core.R import com.terning.core.designsystem.component.item.RadioButtonGroups import com.terning.core.extension.noRippleClickable import com.terning.core.type.ColorType +import timber.log.Timber @Composable fun ColorPalette( - initialColor: ColorType, + selectedColor: ColorType, onColorSelected: (ColorType) -> Unit, modifier: Modifier = Modifier ) { - var selectedColor by remember { mutableStateOf(initialColor) } RadioButtonGroups( options = ColorType.entries, gridCellCount = ColorType.entries.size, - onOptionSelected = { color -> - selectedColor = color - onColorSelected(color) - }, + onOptionSelected = onColorSelected, buttonComposable = { colorType, isSelected, onOptionSelected -> - - ColorButton( colorType = colorType, - isSelected = isSelected,//selectedColor == colorType, + isSelected = selectedColor == colorType, onColorSelected = onOptionSelected ) }, @@ -111,7 +106,7 @@ internal fun ColorButton( @Composable fun ColorPalettePreview() { ColorPalette( - initialColor = ColorType.RED, + selectedColor = ColorType.RED, onColorSelected = {} ) } diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/component/ScrapColorChangeButton.kt b/feature/src/main/java/com/terning/feature/dialog/detail/component/ScrapColorChangeButton.kt new file mode 100644 index 000000000..0850d024d --- /dev/null +++ b/feature/src/main/java/com/terning/feature/dialog/detail/component/ScrapColorChangeButton.kt @@ -0,0 +1,89 @@ +package com.terning.feature.dialog.detail.component + +import androidx.compose.foundation.BorderStroke +import androidx.compose.foundation.interaction.MutableInteractionSource +import androidx.compose.foundation.interaction.collectIsPressedAsState +import androidx.compose.foundation.layout.PaddingValues +import androidx.compose.foundation.layout.fillMaxWidth +import androidx.compose.foundation.layout.wrapContentHeight +import androidx.compose.foundation.shape.RoundedCornerShape +import androidx.compose.material.ripple.LocalRippleTheme +import androidx.compose.material3.Button +import androidx.compose.material3.ButtonDefaults +import androidx.compose.material3.Text +import androidx.compose.runtime.Composable +import androidx.compose.runtime.CompositionLocalProvider +import androidx.compose.runtime.getValue +import androidx.compose.runtime.remember +import androidx.compose.ui.Modifier +import androidx.compose.ui.res.stringResource +import androidx.compose.ui.text.style.TextAlign +import androidx.compose.ui.unit.Dp +import androidx.compose.ui.unit.dp +import com.terning.core.designsystem.theme.Grey150 +import com.terning.core.designsystem.theme.Grey400 +import com.terning.core.designsystem.theme.TerningMain +import com.terning.core.designsystem.theme.TerningSub1 +import com.terning.core.designsystem.theme.TerningSub4 +import com.terning.core.designsystem.theme.TerningSub5 +import com.terning.core.designsystem.theme.TerningTheme +import com.terning.core.designsystem.theme.White +import com.terning.core.util.NoRippleTheme +import com.terning.feature.R + +@Composable +fun ScrapColorChangeButton( + isColorChanged: Boolean, + isColorChangedOnce: Boolean, + cornerRadius: Dp, + paddingVertical: Dp, + onButtonClick: () -> Unit, + modifier: Modifier = Modifier, +) { + val interactionSource = remember { MutableInteractionSource() } + val isPressed by interactionSource.collectIsPressedAsState() + val backgroundColor = when { + isColorChanged && !isPressed -> White + isColorChanged && isPressed -> TerningSub5 + isColorChangedOnce -> TerningSub4 + else -> White + } + val textColor = when { + !isColorChangedOnce && !isColorChanged -> Grey400 + else -> TerningMain + } + val borderColor = when { + !isColorChangedOnce && !isColorChanged -> Grey150 + else -> TerningMain + } + val text = when { + isColorChangedOnce && !isColorChanged -> R.string.dialog_content_calendar_color_change_complete + else -> R.string.dialog_content_calendar_color_change + } + + CompositionLocalProvider(LocalRippleTheme provides NoRippleTheme) { + Button( + contentPadding = PaddingValues(vertical = paddingVertical), + modifier = modifier + .fillMaxWidth() + .wrapContentHeight(), + interactionSource = interactionSource, + colors = ButtonDefaults.buttonColors( + containerColor = backgroundColor, + contentColor = textColor, + ), + border = BorderStroke( + width = 1.dp, + color = borderColor + ), + shape = RoundedCornerShape(cornerRadius), + onClick = { onButtonClick() } + ) { + Text( + text = stringResource(id = text), + style = TerningTheme.typography.button3, + textAlign = TextAlign.Center, + ) + } + } +} diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/state/ScrapDialogUiState.kt b/feature/src/main/java/com/terning/feature/dialog/detail/state/ScrapDialogUiState.kt new file mode 100644 index 000000000..c78c94576 --- /dev/null +++ b/feature/src/main/java/com/terning/feature/dialog/detail/state/ScrapDialogUiState.kt @@ -0,0 +1,10 @@ +package com.terning.feature.dialog.detail.state + +import com.terning.core.type.ColorType + +data class ScrapDialogUiState( + val initialColorType: ColorType = ColorType.RED, + val selectedColorType: ColorType = ColorType.RED, + val isColorChanged: Boolean = false, + val isColorChangedOnce: Boolean = false, +) \ No newline at end of file diff --git a/feature/src/main/res/values/strings.xml b/feature/src/main/res/values/strings.xml index 0190c1a92..46b78e88f 100644 --- a/feature/src/main/res/values/strings.xml +++ b/feature/src/main/res/values/strings.xml @@ -62,6 +62,7 @@ 공고를 캘린더에 스크랩하시겠어요? 스크랩 색상 색상 변경하기 + 색상 변경완료 내 캘린더에 스크랩하기 관심 공고가 캘린더에서 사라져요! 스크랩을 취소하시겠어요? From cd13ab8cd795e3e7367739305bc7af728fadb244 Mon Sep 17 00:00:00 2001 From: boiledegg Date: Tue, 3 Sep 2024 22:45:26 +0900 Subject: [PATCH 15/26] =?UTF-8?q?[DEL/#212]=20=EB=9D=BC=EC=9D=B4=EB=B8=8C?= =?UTF-8?q?=EB=9F=AC=EB=A6=AC=20=EC=A0=95=EB=A6=AC?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- core/src/main/java/com/terning/core/type/ColorType.kt | 1 - .../java/com/terning/feature/dialog/detail/ScrapDialog.kt | 1 - .../terning/feature/dialog/detail/ScrapDialogViewModel.kt | 1 - .../terning/feature/dialog/detail/component/ColorPalette.kt | 5 ----- .../dialog/detail/component/ScrapColorChangeButton.kt | 1 - 5 files changed, 9 deletions(-) diff --git a/core/src/main/java/com/terning/core/type/ColorType.kt b/core/src/main/java/com/terning/core/type/ColorType.kt index 073fc15d5..7d35c8f65 100644 --- a/core/src/main/java/com/terning/core/type/ColorType.kt +++ b/core/src/main/java/com/terning/core/type/ColorType.kt @@ -1,6 +1,5 @@ package com.terning.core.type -import android.util.Log import androidx.compose.ui.graphics.Color import com.terning.core.designsystem.theme.CalBlue2 import com.terning.core.designsystem.theme.CalBlueBc diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt index 24110f0cb..dab00bf86 100644 --- a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt +++ b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt @@ -50,7 +50,6 @@ import com.terning.feature.R import com.terning.feature.dialog.detail.component.ColorPalette import com.terning.feature.dialog.detail.component.ScrapColorChangeButton import com.terning.feature.intern.component.InternInfoRow -import timber.log.Timber @Composable diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogViewModel.kt b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogViewModel.kt index e0510a3f6..8714c912a 100644 --- a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogViewModel.kt +++ b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogViewModel.kt @@ -25,7 +25,6 @@ import kotlinx.coroutines.flow.asSharedFlow import kotlinx.coroutines.flow.asStateFlow import kotlinx.coroutines.flow.update import kotlinx.coroutines.launch -import timber.log.Timber import javax.inject.Inject @HiltViewModel diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/component/ColorPalette.kt b/feature/src/main/java/com/terning/feature/dialog/detail/component/ColorPalette.kt index 13cbdfa0f..cfd6b160d 100644 --- a/feature/src/main/java/com/terning/feature/dialog/detail/component/ColorPalette.kt +++ b/feature/src/main/java/com/terning/feature/dialog/detail/component/ColorPalette.kt @@ -9,10 +9,6 @@ import androidx.compose.foundation.layout.size import androidx.compose.foundation.layout.wrapContentSize import androidx.compose.foundation.shape.CircleShape import androidx.compose.runtime.Composable -import androidx.compose.runtime.getValue -import androidx.compose.runtime.mutableStateOf -import androidx.compose.runtime.remember -import androidx.compose.runtime.setValue import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.graphics.Color @@ -23,7 +19,6 @@ import com.terning.core.R import com.terning.core.designsystem.component.item.RadioButtonGroups import com.terning.core.extension.noRippleClickable import com.terning.core.type.ColorType -import timber.log.Timber @Composable fun ColorPalette( diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/component/ScrapColorChangeButton.kt b/feature/src/main/java/com/terning/feature/dialog/detail/component/ScrapColorChangeButton.kt index 0850d024d..9015a3f77 100644 --- a/feature/src/main/java/com/terning/feature/dialog/detail/component/ScrapColorChangeButton.kt +++ b/feature/src/main/java/com/terning/feature/dialog/detail/component/ScrapColorChangeButton.kt @@ -23,7 +23,6 @@ import androidx.compose.ui.unit.dp import com.terning.core.designsystem.theme.Grey150 import com.terning.core.designsystem.theme.Grey400 import com.terning.core.designsystem.theme.TerningMain -import com.terning.core.designsystem.theme.TerningSub1 import com.terning.core.designsystem.theme.TerningSub4 import com.terning.core.designsystem.theme.TerningSub5 import com.terning.core.designsystem.theme.TerningTheme From c424ad61a83af434d38e625e9f2490267117f337 Mon Sep 17 00:00:00 2001 From: boiledegg Date: Wed, 4 Sep 2024 00:18:07 +0900 Subject: [PATCH 16/26] =?UTF-8?q?[FIX/#212]=20=EB=B2=84=ED=8A=BC=20?= =?UTF-8?q?=EA=B4=80=EB=A0=A8=20=EB=A1=9C=EC=A7=81=20=EC=88=98=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../feature/dialog/detail/ScrapDialog.kt | 44 +++++---------- .../dialog/detail/ScrapDialogSideEffect.kt | 2 +- .../dialog/detail/ScrapDialogViewModel.kt | 56 ++++++++++++------- .../component/ScrapColorChangeButton.kt | 9 ++- .../feature/intern/component/InternInfoRow.kt | 3 +- 5 files changed, 59 insertions(+), 55 deletions(-) diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt index dab00bf86..b52f0d124 100644 --- a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt +++ b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt @@ -76,29 +76,24 @@ fun ScrapDialog( .collect { sideEffect -> when (sideEffect) { is ScrapDialogSideEffect.ShowToast -> {} - is ScrapDialogSideEffect.DismissDialog -> onDismissRequest() - is ScrapDialogSideEffect.ChangedColor -> onClickChangeColor() - is ScrapDialogSideEffect.NavigateToDetail -> onClickNavigateButton( - internshipAnnouncementId - ) - + is ScrapDialogSideEffect.DismissDialog -> { + viewModel.initUiState() + onDismissRequest() + } + is ScrapDialogSideEffect.PatchedScrap -> onClickChangeColor() + is ScrapDialogSideEffect.NavigateToDetail -> onClickNavigateButton(internshipAnnouncementId) is ScrapDialogSideEffect.ScrappedAnnouncement -> {} } } } LaunchedEffect(true) { - with(viewModel) { - val colorType = ColorType.findColorType(scrapColor).takeIf { it != null } - colorType?.let { - updateInitialColorType(colorType) - updateSelectedColorType(colorType) - } - } + val colorType = ColorType.findColorType(scrapColor).takeIf { it != null } + colorType?.let { viewModel.setInitialAndSelectedColorType(it) } } TerningBasicDialog( - onDismissRequest = onDismissRequest + onDismissRequest = viewModel::navigateUp ) { ScrapDialogScreen( title = title, @@ -111,26 +106,13 @@ fun ScrapDialog( selectedColorType = uiState.selectedColorType, isColorChanged = uiState.isColorChanged, isColorChangedOnce = uiState.isColorChangedOnce, - onClickColorButton = { selectedColorType -> - with(viewModel) { - updateSelectedColorType(selectedColorType) - updateIsColorChanged() - } - }, + onClickColorButton = viewModel::changeSelectedColor, onClickColorChangeButton = { - with(viewModel) { - patchScrap(scrapId = scrapId, color = uiState.selectedColorType) - setIsColorChangedOnce() - updateInitialColorType(colorType = uiState.selectedColorType) - } - onClickChangeColor() + viewModel.patchScrap(scrapId = scrapId, color = uiState.selectedColorType) }, onClickNavigateButton = viewModel::navigateToDetail, onClickScrapButton = { - viewModel.postScrap( - internshipAnnouncementId, - uiState.selectedColorType - ) + viewModel.postScrap(internshipAnnouncementId, uiState.selectedColorType) } ) } @@ -215,7 +197,7 @@ private fun ScrapDialogScreen( ) { Text( text = stringResource(id = R.string.dialog_content_color_button), - style = TerningTheme.typography.body5, + style = TerningTheme.typography.body6, color = Grey400, ) } diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogSideEffect.kt b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogSideEffect.kt index e964bf632..a471a583a 100644 --- a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogSideEffect.kt +++ b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogSideEffect.kt @@ -4,6 +4,6 @@ sealed class ScrapDialogSideEffect{ data class ShowToast(val message: Int): ScrapDialogSideEffect() data object DismissDialog : ScrapDialogSideEffect() data object ScrappedAnnouncement : ScrapDialogSideEffect() - data object ChangedColor: ScrapDialogSideEffect() + data object PatchedScrap: ScrapDialogSideEffect() data object NavigateToDetail : ScrapDialogSideEffect() } \ No newline at end of file diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogViewModel.kt b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogViewModel.kt index 8714c912a..0cb9c5dfe 100644 --- a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogViewModel.kt +++ b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogViewModel.kt @@ -37,22 +37,42 @@ class ScrapDialogViewModel @Inject constructor( private var _sideEffect: MutableSharedFlow = MutableSharedFlow() val sideEffect = _sideEffect.asSharedFlow() - fun updateInitialColorType(colorType: ColorType) { + fun initUiState() { _uiState.update { currentState -> - currentState.copy(initialColorType = colorType) + currentState.copy( + isColorChanged = false, + isColorChangedOnce = false + ) } } - fun updateSelectedColorType(colorType: ColorType) { + fun setInitialAndSelectedColorType(colorType: ColorType) { _uiState.update { currentState -> - if (currentState.selectedColorType == colorType) return@update currentState - currentState.copy(selectedColorType = colorType) + currentState.copy( + initialColorType = colorType, + selectedColorType = colorType + ) } } - fun updateIsColorChanged() = _uiState.update { currentState -> - val isColorChanged = _uiState.value.selectedColorType != _uiState.value.initialColorType - currentState.copy(isColorChanged = isColorChanged) + fun changeSelectedColor(colorType: ColorType) { + _uiState.update { currentState -> + currentState.copy( + selectedColorType = colorType, + isColorChanged = colorType != _uiState.value.initialColorType + ) + } + } + + fun navigateToDetail() = viewModelScope.launch { + with(_sideEffect) { + emit(ScrapDialogSideEffect.NavigateToDetail) + emit(ScrapDialogSideEffect.DismissDialog) + } + } + + fun navigateUp() = viewModelScope.launch { + _sideEffect.emit(ScrapDialogSideEffect.DismissDialog) } fun postScrap(id: Long, color: ColorType) { @@ -74,23 +94,19 @@ class ScrapDialogViewModel @Inject constructor( val colorIndex = getColorIndex(color.main) scrapRepository.patchScrap(CalendarScrapRequest(scrapId, colorIndex)) .onSuccess { - _sideEffect.emit(ScrapDialogSideEffect.ChangedColor) - + _sideEffect.emit(ScrapDialogSideEffect.PatchedScrap) + _uiState.update { currentState -> + currentState.copy( + initialColorType = color, + isColorChanged = (_uiState.value.selectedColorType != color), + isColorChangedOnce = true, + ) + } }.onFailure { _sideEffect.emit(ScrapDialogSideEffect.ShowToast(R.string.server_failure)) } } - fun setIsColorChangedOnce() = _uiState.update { currentState -> - if (!currentState.isColorChangedOnce) return@update currentState - currentState.copy(isColorChangedOnce = true) - } - - fun navigateToDetail() = viewModelScope.launch { - _sideEffect.emit(ScrapDialogSideEffect.NavigateToDetail) - } - - private fun getColorIndex(color: Color): Int = listOf( CalRed, CalOrange1, diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/component/ScrapColorChangeButton.kt b/feature/src/main/java/com/terning/feature/dialog/detail/component/ScrapColorChangeButton.kt index 9015a3f77..d353d1ed4 100644 --- a/feature/src/main/java/com/terning/feature/dialog/detail/component/ScrapColorChangeButton.kt +++ b/feature/src/main/java/com/terning/feature/dialog/detail/component/ScrapColorChangeButton.kt @@ -29,6 +29,7 @@ import com.terning.core.designsystem.theme.TerningTheme import com.terning.core.designsystem.theme.White import com.terning.core.util.NoRippleTheme import com.terning.feature.R +import timber.log.Timber @Composable fun ScrapColorChangeButton( @@ -41,10 +42,14 @@ fun ScrapColorChangeButton( ) { val interactionSource = remember { MutableInteractionSource() } val isPressed by interactionSource.collectIsPressedAsState() + + Timber.tag("ScrapDialog") + .d("isColorChanged: $isColorChanged\nisColorChangedOnce: $isColorChangedOnce\nisPressed: $isPressed") + val backgroundColor = when { + isColorChangedOnce && !isColorChanged -> TerningSub4 isColorChanged && !isPressed -> White isColorChanged && isPressed -> TerningSub5 - isColorChangedOnce -> TerningSub4 else -> White } val textColor = when { @@ -76,7 +81,7 @@ fun ScrapColorChangeButton( color = borderColor ), shape = RoundedCornerShape(cornerRadius), - onClick = { onButtonClick() } + onClick = onButtonClick ) { Text( text = stringResource(id = text), diff --git a/feature/src/main/java/com/terning/feature/intern/component/InternInfoRow.kt b/feature/src/main/java/com/terning/feature/intern/component/InternInfoRow.kt index b01654710..34d23c16f 100644 --- a/feature/src/main/java/com/terning/feature/intern/component/InternInfoRow.kt +++ b/feature/src/main/java/com/terning/feature/intern/component/InternInfoRow.kt @@ -7,6 +7,7 @@ import androidx.compose.runtime.Composable import androidx.compose.ui.Alignment import androidx.compose.ui.unit.dp import com.terning.core.designsystem.theme.Black +import com.terning.core.designsystem.theme.Grey400 import com.terning.core.designsystem.theme.Grey500 import com.terning.core.designsystem.theme.TerningTheme @@ -19,7 +20,7 @@ fun InternInfoRow(title: String, value: String) { Text( text = title, style = TerningTheme.typography.body2, - color = Black, + color = Grey400, ) Text( text = value, From 40156b67a741dbe0f11a047008f243f2022fffde Mon Sep 17 00:00:00 2001 From: boiledegg Date: Wed, 4 Sep 2024 00:45:40 +0900 Subject: [PATCH 17/26] =?UTF-8?q?[UI/#212]=20=EC=8A=A4=ED=81=AC=EB=9E=A9?= =?UTF-8?q?=EC=B0=BD=20=EA=B8=B0=ED=83=80=20UI=20=EC=88=98=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../component/dialog/TerningBasicDialog.kt | 3 +- .../com/terning/core/extension/TextStyle.kt | 11 ++++ .../feature/dialog/detail/ScrapDialog.kt | 63 ++++++++++--------- .../feature/intern/component/InternInfoRow.kt | 3 +- 4 files changed, 49 insertions(+), 31 deletions(-) create mode 100644 core/src/main/java/com/terning/core/extension/TextStyle.kt diff --git a/core/src/main/java/com/terning/core/designsystem/component/dialog/TerningBasicDialog.kt b/core/src/main/java/com/terning/core/designsystem/component/dialog/TerningBasicDialog.kt index 2264c1cf0..ab1df8a46 100644 --- a/core/src/main/java/com/terning/core/designsystem/component/dialog/TerningBasicDialog.kt +++ b/core/src/main/java/com/terning/core/designsystem/component/dialog/TerningBasicDialog.kt @@ -58,7 +58,8 @@ fun TerningBasicDialog( Row( modifier = Modifier .fillMaxWidth() - .padding(18.dp), + .padding(top = 18.dp, bottom = 16.dp) + .padding(horizontal = 16.dp), horizontalArrangement = Arrangement.End ) { Icon( diff --git a/core/src/main/java/com/terning/core/extension/TextStyle.kt b/core/src/main/java/com/terning/core/extension/TextStyle.kt new file mode 100644 index 000000000..b72217e3d --- /dev/null +++ b/core/src/main/java/com/terning/core/extension/TextStyle.kt @@ -0,0 +1,11 @@ +package com.terning.core.extension + +import androidx.compose.runtime.Composable +import androidx.compose.ui.platform.LocalDensity +import androidx.compose.ui.text.TextStyle +import androidx.compose.ui.unit.Dp + +@Composable +fun TextStyle.getFixHeightByMaxLine(maxLine: Int): Dp = with(LocalDensity.current) { + lineHeight.toDp() * maxLine +} diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt index b52f0d124..7d55e0a74 100644 --- a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt +++ b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt @@ -27,6 +27,7 @@ import androidx.compose.ui.layout.ContentScale import androidx.compose.ui.platform.LocalContext import androidx.compose.ui.res.stringResource import androidx.compose.ui.text.style.TextAlign +import androidx.compose.ui.text.style.TextOverflow import androidx.compose.ui.tooling.preview.Preview import androidx.compose.ui.unit.dp import androidx.hilt.navigation.compose.hiltViewModel @@ -39,12 +40,12 @@ import com.terning.core.designsystem.component.button.RoundButton import com.terning.core.designsystem.component.dialog.TerningBasicDialog import com.terning.core.designsystem.theme.Grey100 import com.terning.core.designsystem.theme.Grey200 -import com.terning.core.designsystem.theme.Grey350 import com.terning.core.designsystem.theme.Grey400 import com.terning.core.designsystem.theme.Grey500 import com.terning.core.designsystem.theme.TerningMain import com.terning.core.designsystem.theme.TerningPointTheme import com.terning.core.designsystem.theme.TerningTheme +import com.terning.core.extension.getFixHeightByMaxLine import com.terning.core.type.ColorType import com.terning.feature.R import com.terning.feature.dialog.detail.component.ColorPalette @@ -80,8 +81,12 @@ fun ScrapDialog( viewModel.initUiState() onDismissRequest() } + is ScrapDialogSideEffect.PatchedScrap -> onClickChangeColor() - is ScrapDialogSideEffect.NavigateToDetail -> onClickNavigateButton(internshipAnnouncementId) + is ScrapDialogSideEffect.NavigateToDetail -> onClickNavigateButton( + internshipAnnouncementId + ) + is ScrapDialogSideEffect.ScrappedAnnouncement -> {} } } @@ -163,22 +168,25 @@ private fun ScrapDialogScreen( shape = RoundedCornerShape(size = 15.dp) ) ) - Text( - text = title, - textAlign = TextAlign.Center, - style = TerningTheme.typography.title4, - color = Grey500, - modifier = Modifier.padding(top = 20.dp) - ) - Text( - text = stringResource(id = R.string.dialog_scrap_mine), - style = TerningTheme.typography.body5, - color = Grey350, - modifier = Modifier.padding( - top = 4.dp + + Spacer(modifier = Modifier.height(15.dp)) + Box( + modifier = Modifier + .height(TerningTheme.typography.title4.getFixHeightByMaxLine(3)) + .fillMaxWidth(), + contentAlignment = Alignment.Center + ) { + Text( + text = title, + textAlign = TextAlign.Center, + style = TerningTheme.typography.title4, + color = Grey500, + overflow = TextOverflow.Ellipsis, ) - ) - Spacer(modifier = Modifier.height(26.dp)) + } + + Spacer(modifier = Modifier.height(10.dp)) + Column( horizontalAlignment = Alignment.Start, verticalArrangement = Arrangement.Top, @@ -202,13 +210,10 @@ private fun ScrapDialogScreen( ) } + Spacer(modifier = Modifier.height(8.dp)) + Box( - modifier = Modifier - .fillMaxWidth() - .padding( - top = 12.dp, - bottom = 23.dp, - ), + modifier = Modifier.fillMaxWidth(), contentAlignment = Alignment.Center ) { ColorPalette( @@ -219,12 +224,12 @@ private fun ScrapDialogScreen( HorizontalDivider( thickness = 1.dp, color = Grey200, - modifier = Modifier.padding( - bottom = 8.dp - ) + modifier = Modifier.padding(vertical = 8.dp) ) + + Column( - modifier = Modifier.padding(bottom = 29.dp), + modifier = Modifier.padding(vertical = 4.dp), verticalArrangement = Arrangement.spacedBy( 5.dp, Alignment.CenterVertically @@ -246,6 +251,8 @@ private fun ScrapDialogScreen( } } + Spacer(modifier = Modifier.height(8.dp)) + if (isScrapped) { DetailScrapButton( isColorChanged = isColorChanged, @@ -312,7 +319,7 @@ private fun DetailScrapButton( private fun ScrapDialogPreview() { TerningPointTheme { ScrapDialogScreen( - title = "터닝 하반기 채용", + title = "터닝 하반기 채용터닝 하반기 채용터닝 하반기 채용터닝터닝 하반기 채용터닝터닝 하반기 채용터닝터닝 하반기 채용터닝터닝 하반기 채용터닝터닝 하반기 채용터닝 하반기 채용터닝 하반기 채용터닝 하반기 채용터닝 하반기 채용터닝 하반기 채용터닝 하반기 채용", deadline = "2024/09/07", startYear = 2024, startMonth = 11, diff --git a/feature/src/main/java/com/terning/feature/intern/component/InternInfoRow.kt b/feature/src/main/java/com/terning/feature/intern/component/InternInfoRow.kt index 34d23c16f..073e4ae7b 100644 --- a/feature/src/main/java/com/terning/feature/intern/component/InternInfoRow.kt +++ b/feature/src/main/java/com/terning/feature/intern/component/InternInfoRow.kt @@ -6,7 +6,6 @@ import androidx.compose.material3.Text import androidx.compose.runtime.Composable import androidx.compose.ui.Alignment import androidx.compose.ui.unit.dp -import com.terning.core.designsystem.theme.Black import com.terning.core.designsystem.theme.Grey400 import com.terning.core.designsystem.theme.Grey500 import com.terning.core.designsystem.theme.TerningTheme @@ -25,7 +24,7 @@ fun InternInfoRow(title: String, value: String) { Text( text = value, style = TerningTheme.typography.body3, - color = Grey500, + color = Grey400, ) } } \ No newline at end of file From dd3d48811e786f813ccea4f5d1913f5f3c05fdd2 Mon Sep 17 00:00:00 2001 From: boiledegg Date: Wed, 4 Sep 2024 01:00:12 +0900 Subject: [PATCH 18/26] =?UTF-8?q?[UI/#212]=20=EC=B7=A8=EC=86=8C=EC=B0=BD?= =?UTF-8?q?=20=EA=B8=B0=ED=83=80=20UI=20=EC=88=98=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../dialog/cancel/ScrapCancelDialog.kt | 23 +++++++++---------- 1 file changed, 11 insertions(+), 12 deletions(-) diff --git a/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt index 8fb590437..a8ceed4b3 100644 --- a/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt +++ b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt @@ -7,6 +7,7 @@ import androidx.compose.foundation.layout.aspectRatio import androidx.compose.foundation.layout.fillMaxWidth import androidx.compose.foundation.layout.height import androidx.compose.foundation.layout.padding +import androidx.compose.foundation.layout.wrapContentHeight import androidx.compose.material3.Text import androidx.compose.runtime.Composable import androidx.compose.runtime.LaunchedEffect @@ -62,15 +63,11 @@ private fun ScrapCancelScreen( TerningBasicDialog( onDismissRequest = onDismissRequest ) { - Box( - modifier = Modifier - .fillMaxWidth() - .padding(top = 50.dp), - contentAlignment = Alignment.TopCenter - ) { Column( modifier = Modifier .fillMaxWidth() + .wrapContentHeight() + .padding(top = 60.dp) .padding(horizontal = 16.dp), horizontalAlignment = Alignment.CenterHorizontally ) { @@ -78,22 +75,24 @@ private fun ScrapCancelScreen( jsonString = "terning_scrap_cancel.json", modifier = Modifier .fillMaxWidth() - .aspectRatio(1f) + .height(203.dp) ) + Spacer(modifier = Modifier.height(20.dp)) + Text( text = stringResource(id = R.string.dialog_content_scrap_cancel_main_title), textAlign = TextAlign.Center, style = TerningTheme.typography.title4, color = Grey500, ) + + Spacer(modifier = Modifier.height(5.dp)) + Text( text = stringResource(id = R.string.dialog_content_scrap_cancel_sub_title), style = TerningTheme.typography.body5, - color = Grey350, - modifier = Modifier.padding( - top = 5.dp - ) + color = Grey350 ) Spacer(modifier = Modifier.height(40.dp)) @@ -106,7 +105,7 @@ private fun ScrapCancelScreen( ) Spacer(modifier = Modifier.height(16.dp)) } - } + } } From 6f91c700c7956f13d320d398480c03d2d4fd133c Mon Sep 17 00:00:00 2001 From: boiledegg Date: Wed, 4 Sep 2024 01:00:45 +0900 Subject: [PATCH 19/26] =?UTF-8?q?[DEL/#212]=20=EB=9D=BC=EC=9D=B4=EB=B8=8C?= =?UTF-8?q?=EB=9F=AC=EB=A6=AC=20=EC=A0=95=EB=A6=AC?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt | 2 -- 1 file changed, 2 deletions(-) diff --git a/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt index a8ceed4b3..e5ab4b317 100644 --- a/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt +++ b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt @@ -1,9 +1,7 @@ package com.terning.feature.dialog.cancel -import androidx.compose.foundation.layout.Box import androidx.compose.foundation.layout.Column import androidx.compose.foundation.layout.Spacer -import androidx.compose.foundation.layout.aspectRatio import androidx.compose.foundation.layout.fillMaxWidth import androidx.compose.foundation.layout.height import androidx.compose.foundation.layout.padding From 222a636e6d04cc212623d963620d2e8d348ce778 Mon Sep 17 00:00:00 2001 From: boiledegg Date: Wed, 4 Sep 2024 01:13:27 +0900 Subject: [PATCH 20/26] =?UTF-8?q?[FIX/#212]=20=EC=8A=A4=ED=81=AC=EB=9E=A9?= =?UTF-8?q?=EC=88=98=EC=A0=95=20=EB=B2=84=ED=8A=BC=20=EC=98=A4=EB=A5=98=20?= =?UTF-8?q?=ED=95=B4=EA=B2=B0?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../main/java/com/terning/feature/dialog/detail/ScrapDialog.kt | 1 + .../com/terning/feature/dialog/detail/ScrapDialogViewModel.kt | 2 +- 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt index 7d55e0a74..7e7c304ad 100644 --- a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt +++ b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt @@ -113,6 +113,7 @@ fun ScrapDialog( isColorChangedOnce = uiState.isColorChangedOnce, onClickColorButton = viewModel::changeSelectedColor, onClickColorChangeButton = { + if(uiState.isColorChanged) viewModel.patchScrap(scrapId = scrapId, color = uiState.selectedColorType) }, onClickNavigateButton = viewModel::navigateToDetail, diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogViewModel.kt b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogViewModel.kt index 0cb9c5dfe..9e2de8d01 100644 --- a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogViewModel.kt +++ b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogViewModel.kt @@ -98,7 +98,7 @@ class ScrapDialogViewModel @Inject constructor( _uiState.update { currentState -> currentState.copy( initialColorType = color, - isColorChanged = (_uiState.value.selectedColorType != color), + isColorChanged = false, isColorChangedOnce = true, ) } From ddc1cc1f391a62ecb4ce3f11b535bbc4564f98dd Mon Sep 17 00:00:00 2001 From: boiledegg Date: Wed, 4 Sep 2024 01:34:11 +0900 Subject: [PATCH 21/26] =?UTF-8?q?[DEL/#212]=20=EB=A1=9C=EA=B7=B8=20?= =?UTF-8?q?=EC=82=AD=EC=A0=9C?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../feature/dialog/detail/component/ScrapColorChangeButton.kt | 3 --- 1 file changed, 3 deletions(-) diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/component/ScrapColorChangeButton.kt b/feature/src/main/java/com/terning/feature/dialog/detail/component/ScrapColorChangeButton.kt index d353d1ed4..bc322247d 100644 --- a/feature/src/main/java/com/terning/feature/dialog/detail/component/ScrapColorChangeButton.kt +++ b/feature/src/main/java/com/terning/feature/dialog/detail/component/ScrapColorChangeButton.kt @@ -43,9 +43,6 @@ fun ScrapColorChangeButton( val interactionSource = remember { MutableInteractionSource() } val isPressed by interactionSource.collectIsPressedAsState() - Timber.tag("ScrapDialog") - .d("isColorChanged: $isColorChanged\nisColorChangedOnce: $isColorChangedOnce\nisPressed: $isPressed") - val backgroundColor = when { isColorChangedOnce && !isColorChanged -> TerningSub4 isColorChanged && !isPressed -> White From 536c53c47e1f720553767037a82c878fdb52d195 Mon Sep 17 00:00:00 2001 From: boiledegg Date: Wed, 4 Sep 2024 01:34:39 +0900 Subject: [PATCH 22/26] =?UTF-8?q?[DEL/#212]=20=EB=9D=BC=EC=9D=B4=EB=B8=8C?= =?UTF-8?q?=EB=9F=AC=EB=A6=AC=20=EC=82=AD=EC=A0=9C?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../feature/dialog/detail/component/ScrapColorChangeButton.kt | 1 - 1 file changed, 1 deletion(-) diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/component/ScrapColorChangeButton.kt b/feature/src/main/java/com/terning/feature/dialog/detail/component/ScrapColorChangeButton.kt index bc322247d..59fe57c74 100644 --- a/feature/src/main/java/com/terning/feature/dialog/detail/component/ScrapColorChangeButton.kt +++ b/feature/src/main/java/com/terning/feature/dialog/detail/component/ScrapColorChangeButton.kt @@ -29,7 +29,6 @@ import com.terning.core.designsystem.theme.TerningTheme import com.terning.core.designsystem.theme.White import com.terning.core.util.NoRippleTheme import com.terning.feature.R -import timber.log.Timber @Composable fun ScrapColorChangeButton( From 05ffbe0cc20f72dbdb4b9c204d0b4abb3fa9c52d Mon Sep 17 00:00:00 2001 From: boiledegg Date: Wed, 4 Sep 2024 22:00:41 +0900 Subject: [PATCH 23/26] =?UTF-8?q?[REFACTOR/#212]=20=EB=A6=AC=EB=B7=B0=20?= =?UTF-8?q?=EA=B8=B0=EB=B0=98=20=EC=BD=94=EB=93=9C=20=EC=88=98=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../component/dialog/TerningBasicDialog.kt | 10 ---------- .../feature/dialog/detail/component/ColorPalette.kt | 3 +-- 2 files changed, 1 insertion(+), 12 deletions(-) diff --git a/core/src/main/java/com/terning/core/designsystem/component/dialog/TerningBasicDialog.kt b/core/src/main/java/com/terning/core/designsystem/component/dialog/TerningBasicDialog.kt index ab1df8a46..2adeeaaba 100644 --- a/core/src/main/java/com/terning/core/designsystem/component/dialog/TerningBasicDialog.kt +++ b/core/src/main/java/com/terning/core/designsystem/component/dialog/TerningBasicDialog.kt @@ -4,30 +4,20 @@ import androidx.compose.foundation.background import androidx.compose.foundation.layout.Arrangement import androidx.compose.foundation.layout.Box 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.height import androidx.compose.foundation.layout.padding -import androidx.compose.foundation.layout.size -import androidx.compose.foundation.layout.wrapContentHeight import androidx.compose.foundation.layout.wrapContentSize import androidx.compose.foundation.shape.RoundedCornerShape import androidx.compose.material3.Icon -import androidx.compose.material3.IconButton import androidx.compose.runtime.Composable -import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.res.painterResource import androidx.compose.ui.tooling.preview.Preview -import androidx.compose.ui.unit.Constraints import androidx.compose.ui.unit.dp import androidx.compose.ui.window.Dialog import androidx.compose.ui.window.DialogProperties import com.terning.core.R -import com.terning.core.designsystem.component.button.RoundButton import com.terning.core.designsystem.theme.Grey300 -import com.terning.core.designsystem.theme.TerningTheme import com.terning.core.designsystem.theme.White import com.terning.core.extension.noRippleClickable diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/component/ColorPalette.kt b/feature/src/main/java/com/terning/feature/dialog/detail/component/ColorPalette.kt index cfd6b160d..f752e464c 100644 --- a/feature/src/main/java/com/terning/feature/dialog/detail/component/ColorPalette.kt +++ b/feature/src/main/java/com/terning/feature/dialog/detail/component/ColorPalette.kt @@ -26,7 +26,6 @@ fun ColorPalette( onColorSelected: (ColorType) -> Unit, modifier: Modifier = Modifier ) { - RadioButtonGroups( options = ColorType.entries, gridCellCount = ColorType.entries.size, @@ -90,7 +89,7 @@ internal fun ColorButton( if (isSelected) { Image( painter = painterResource(id = R.drawable.ic_color_check), - contentDescription = "" + contentDescription = "color selected" ) } } From 49b2275da9afd78101d2557dfe243f2155eccb36 Mon Sep 17 00:00:00 2001 From: boiledegg Date: Wed, 4 Sep 2024 22:00:59 +0900 Subject: [PATCH 24/26] =?UTF-8?q?[DOC/#212]=20RadioButtonGroups=20?= =?UTF-8?q?=EB=AC=B8=EC=84=9C=ED=99=94?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../designsystem/component/item/RadioButtonGroups.kt | 12 ++++++++++++ 1 file changed, 12 insertions(+) diff --git a/core/src/main/java/com/terning/core/designsystem/component/item/RadioButtonGroups.kt b/core/src/main/java/com/terning/core/designsystem/component/item/RadioButtonGroups.kt index 3ed2d3919..13d28d93b 100644 --- a/core/src/main/java/com/terning/core/designsystem/component/item/RadioButtonGroups.kt +++ b/core/src/main/java/com/terning/core/designsystem/component/item/RadioButtonGroups.kt @@ -12,6 +12,18 @@ import androidx.compose.runtime.setValue import androidx.compose.ui.Modifier import androidx.compose.ui.unit.Dp +/** + * 라디오버튼 그룹을 관리하기 위한 컴포넌트입니다. + * + * @param options 라디오버튼에 표시될 옵션 목록입니다. + * @param buttonComposable 라디오버튼을 표시하는 컴포저블입니다. + * @param gridCellCount Row 하나에 들어갈 버튼 개수입니다. + * @param verticalArrangementSpace Row 간격입니다. + * @param horizontalArrangementSpace Column 간격입니다. + * @param modifier 수정자입니다. + * @param onOptionSelected 옵션이 선택되었을 때 호출되는 콜백입니다. + */ + @Composable fun RadioButtonGroups( options: List, From 0eac245133e69435a6f56b390f8a3ab82ed25bd6 Mon Sep 17 00:00:00 2001 From: boiledegg Date: Wed, 4 Sep 2024 22:09:26 +0900 Subject: [PATCH 25/26] =?UTF-8?q?[CHORE/#212]=20=EB=A6=AC=EC=86=8C?= =?UTF-8?q?=EC=8A=A4=20=EC=A3=BC=EC=84=9D=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../terning/feature/dialog/cancel/ScrapCancelSideEffect.kt | 4 +++- .../terning/feature/dialog/detail/ScrapDialogSideEffect.kt | 4 +++- 2 files changed, 6 insertions(+), 2 deletions(-) diff --git a/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelSideEffect.kt b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelSideEffect.kt index c91071090..351729eaf 100644 --- a/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelSideEffect.kt +++ b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelSideEffect.kt @@ -1,6 +1,8 @@ package com.terning.feature.dialog.cancel +import androidx.annotation.StringRes + sealed class ScrapCancelSideEffect{ - data class ShowToast(val message: Int): ScrapCancelSideEffect() + data class ShowToast(@StringRes val message: Int): ScrapCancelSideEffect() data object DismissDialog : ScrapCancelSideEffect() } \ No newline at end of file diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogSideEffect.kt b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogSideEffect.kt index a471a583a..67343309b 100644 --- a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogSideEffect.kt +++ b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogSideEffect.kt @@ -1,7 +1,9 @@ package com.terning.feature.dialog.detail +import androidx.annotation.StringRes + sealed class ScrapDialogSideEffect{ - data class ShowToast(val message: Int): ScrapDialogSideEffect() + data class ShowToast(@StringRes val message: Int): ScrapDialogSideEffect() data object DismissDialog : ScrapDialogSideEffect() data object ScrappedAnnouncement : ScrapDialogSideEffect() data object PatchedScrap: ScrapDialogSideEffect() From 571e43e2d21a11ea6ef2fbdb19b2602ba493c66d Mon Sep 17 00:00:00 2001 From: boiledegg Date: Thu, 5 Sep 2024 01:29:57 +0900 Subject: [PATCH 26/26] =?UTF-8?q?[UI/#212]=20=ED=86=A0=EC=8A=A4=ED=8A=B8?= =?UTF-8?q?=EB=A9=94=EC=8B=9C=EC=A7=80=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../terning/feature/dialog/cancel/ScrapCancelDialog.kt | 7 ++++++- .../feature/dialog/cancel/ScrapCancelViewModel.kt | 6 +++++- .../com/terning/feature/dialog/detail/ScrapDialog.kt | 9 +++++++-- .../feature/dialog/detail/ScrapDialogViewModel.kt | 2 +- feature/src/main/res/values/strings.xml | 2 ++ 5 files changed, 21 insertions(+), 5 deletions(-) diff --git a/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt index e5ab4b317..3e3c8fd77 100644 --- a/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt +++ b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt @@ -11,6 +11,7 @@ import androidx.compose.runtime.Composable import androidx.compose.runtime.LaunchedEffect import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier +import androidx.compose.ui.platform.LocalContext import androidx.compose.ui.res.stringResource import androidx.compose.ui.text.style.TextAlign import androidx.compose.ui.tooling.preview.Preview @@ -25,6 +26,7 @@ import com.terning.core.designsystem.theme.Grey350 import com.terning.core.designsystem.theme.Grey500 import com.terning.core.designsystem.theme.TerningPointTheme import com.terning.core.designsystem.theme.TerningTheme +import com.terning.core.extension.toast import com.terning.feature.R @Composable @@ -33,6 +35,7 @@ fun ScrapCancelDialog( onDismissRequest: (Boolean) -> Unit, viewModel: ScrapCancelViewModel = hiltViewModel() ) { + val context = LocalContext.current val lifecycleOwner = LocalLifecycleOwner.current LaunchedEffect(viewModel.sideEffect, lifecycleOwner) { viewModel.sideEffect.flowWithLifecycle(lifecycle = lifecycleOwner.lifecycle) @@ -41,7 +44,9 @@ fun ScrapCancelDialog( is ScrapCancelSideEffect.DismissDialog -> { onDismissRequest(true) } - is ScrapCancelSideEffect.ShowToast -> {} + is ScrapCancelSideEffect.ShowToast -> { + context.toast(sideEffect.message) + } } } } diff --git a/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelViewModel.kt b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelViewModel.kt index beb35c20d..0cf47d3a2 100644 --- a/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelViewModel.kt +++ b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelViewModel.kt @@ -23,7 +23,11 @@ class ScrapCancelViewModel @Inject constructor( ) = viewModelScope.launch { scrapRepository.deleteScrap(CalendarScrapRequest(scrapId, null)) .onSuccess { - _sideEffect.emit(ScrapCancelSideEffect.DismissDialog) + with(_sideEffect){ + emit(ScrapCancelSideEffect.DismissDialog) + emit(ScrapCancelSideEffect.ShowToast(R.string.dialog_scrap_cancelled)) + } + }.onFailure { _sideEffect.emit(ScrapCancelSideEffect.ShowToast(R.string.server_failure)) } diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt index 7e7c304ad..e160aea7a 100644 --- a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt +++ b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt @@ -46,6 +46,7 @@ import com.terning.core.designsystem.theme.TerningMain import com.terning.core.designsystem.theme.TerningPointTheme import com.terning.core.designsystem.theme.TerningTheme import com.terning.core.extension.getFixHeightByMaxLine +import com.terning.core.extension.toast import com.terning.core.type.ColorType import com.terning.feature.R import com.terning.feature.dialog.detail.component.ColorPalette @@ -70,13 +71,16 @@ fun ScrapDialog( onClickNavigateButton: (Long) -> Unit, viewModel: ScrapDialogViewModel = hiltViewModel() ) { + val context = LocalContext.current val lifecycleOwner = LocalLifecycleOwner.current val uiState by viewModel.uiState.collectAsStateWithLifecycle(lifecycleOwner = lifecycleOwner) LaunchedEffect(viewModel.sideEffect, lifecycleOwner) { viewModel.sideEffect.flowWithLifecycle(lifecycle = lifecycleOwner.lifecycle) .collect { sideEffect -> when (sideEffect) { - is ScrapDialogSideEffect.ShowToast -> {} + is ScrapDialogSideEffect.ShowToast -> { + context.toast(sideEffect.message) + } is ScrapDialogSideEffect.DismissDialog -> { viewModel.initUiState() onDismissRequest() @@ -87,7 +91,8 @@ fun ScrapDialog( internshipAnnouncementId ) - is ScrapDialogSideEffect.ScrappedAnnouncement -> {} + is ScrapDialogSideEffect.ScrappedAnnouncement -> { + } } } } diff --git a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogViewModel.kt b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogViewModel.kt index 9e2de8d01..3f08e2e4b 100644 --- a/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogViewModel.kt +++ b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogViewModel.kt @@ -81,7 +81,7 @@ class ScrapDialogViewModel @Inject constructor( scrapRepository.postScrap(CalendarScrapRequest(id, colorIndex)) .onSuccess { with(_sideEffect) { - emit(ScrapDialogSideEffect.ShowToast(R.string.intern_scrap_add_toast_message)) + emit(ScrapDialogSideEffect.ShowToast(R.string.dialog_scrap_scrapped)) emit(ScrapDialogSideEffect.ScrappedAnnouncement) } }.onFailure { diff --git a/feature/src/main/res/values/strings.xml b/feature/src/main/res/values/strings.xml index 46b78e88f..9f970dce3 100644 --- a/feature/src/main/res/values/strings.xml +++ b/feature/src/main/res/values/strings.xml @@ -70,6 +70,8 @@ 내가 스크랩한 관심 공고에요! 공고 상세 정보 보기 오늘 마감되는 공고예요! + 관심 공고가 캘린더에 스크랩되었어요! + 관심 공고가 캘린더에서 사라졌어요! 오늘 마감되는 %s님의 관심 공고