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..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 @@ -58,7 +48,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/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, 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 807bc9813..38e375622 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 @@ -42,6 +42,22 @@ val CalBlue2 = Color(0xFF4AA9F2) val CalPurple = Color(0xFF9B64E2) val CalPink = Color(0xFFF260AC) +// 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) val SundayRed = Color(0xFFEB1211) 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/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..7d35c8f65 --- /dev/null +++ b/core/src/main/java/com/terning/core/type/ColorType.kt @@ -0,0 +1,38 @@ +package com.terning.core.type + +import androidx.compose.ui.graphics.Color +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( + val main: Color, + val border: Color, + val sub: Color +) { + 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 diff --git a/core/src/main/res/values/strings.xml b/core/src/main/res/values/strings.xml index a8a7b28f4..2a3dd454b 100644 --- a/core/src/main/res/values/strings.xml +++ b/core/src/main/res/values/strings.xml @@ -43,14 +43,14 @@ 공고를 캘린더에 스크랩하시겠어요? - 색상 - 색상 저장하기 + 스크랩 색상 + 색상 변경하기 내 캘린더에 스크랩하기 관심 공고가 캘린더에서 사라져요! 스크랩을 취소하시겠어요? 스크랩 취소하기 내가 스크랩한 관심 공고에요! - 공고 상세 정보 보러가기 + 공고 상세 정보 보기 오늘 마감되는 공고예요! 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/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 d5563b7ff..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,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 com.terning.feature.dialog.detail.ScrapDialog import kotlinx.coroutines.flow.distinctUntilChanged import java.time.LocalDate @@ -85,18 +85,29 @@ fun CalendarListRoute( listState = listState, uiState = uiState, modifier = modifier, - navigateToAnnouncement = navigateToAnnouncement, - onDismissCancelDialog = { viewModel.updateScrapCancelDialogVisibility(false) }, + 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) }, - onClickScrapCancel = { uiState.scrapId?.let { viewModel.deleteScrap(it) } }, + onClickChangeColor = { + viewModel.getScrapMonthList(uiState.currentDate) + }, 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 +118,9 @@ private fun CalendarListScreen( listState: LazyListState, uiState: CalendarListUiState, navigateToAnnouncement: (Long) -> Unit, - onDismissCancelDialog: () -> Unit, + onDismissCancelDialog: (Boolean) -> Unit, onDismissInternDialog: () -> Unit, - onClickChangeColor: (Color) -> Unit, - onClickScrapCancel: () -> Unit, + onClickChangeColor: () -> Unit, onClickInternship: (CalendarScrapDetail) -> Unit, onClickScrapButton: (Long) -> Unit, modifier: Modifier = Modifier @@ -189,26 +199,37 @@ private fun CalendarListScreen( } if (uiState.scrapDialogVisibility) { - CalendarCancelDialog( - onDismissRequest = onDismissCancelDialog, - onClickScrapCancel = { - onClickScrapCancel() - onDismissCancelDialog() - } - ) + uiState.scrapId?.run { + ScrapCancelDialog( + scrapId = this, + onDismissRequest = onDismissCancelDialog + ) + } } 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/list/CalendarListViewModel.kt b/feature/src/main/java/com/terning/feature/calendar/list/CalendarListViewModel.kt index a178a7f4d..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 @@ -34,8 +21,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 +96,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 44ee9d65c..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 @@ -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 @@ -33,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 com.terning.feature.dialog.detail.ScrapDialog import okhttp3.internal.toImmutableList import java.time.LocalDate @@ -76,12 +81,19 @@ fun CalendarWeekRoute( uiState = uiState, selectedDate = calendarUiState.selectedDate, updateSelectedDate = updateSelectedDate, - navigateToAnnouncement = navigateToAnnouncement, - onDismissCancelDialog = { viewModel.updateScrapCancelDialogVisibility(false) }, + navigateToAnnouncement = { announcementId -> + navigateToAnnouncement(announcementId) + viewModel.updateInternDialogVisibility(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 -> + onClickChangeColor = { viewModel.getScrapWeekList(uiState.selectedDate) }, + onClickScrapButton = { scrapId -> with(viewModel) { updateScrapId(scrapId) updateScrapCancelDialogVisibility(true) @@ -101,21 +113,51 @@ private fun CalendarWeekScreen( uiState: CalendarWeekUiState, selectedDate: LocalDate, updateSelectedDate: (LocalDate) -> Unit, - onDismissCancelDialog: () -> Unit, + onDismissCancelDialog: (Boolean) -> Unit, onDismissInternDialog: () -> Unit, - onClickChangeColor: (Color) -> Unit, - onClickScrapCancel: () -> Unit, + onClickChangeColor: () -> Unit, onClickInternship: (CalendarScrapDetail) -> Unit, onClickScrapButton: (Long) -> Unit, 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 @@ -137,6 +179,7 @@ private fun CalendarWeekScreen( is UiState.Empty -> { CalendarWeekEmpty() } + is UiState.Failure -> {} is UiState.Success -> { CalendarWeekSuccess( @@ -150,26 +193,37 @@ private fun CalendarWeekScreen( } if (uiState.scrapDialogVisibility) { - CalendarCancelDialog( - onDismissRequest = onDismissCancelDialog, - onClickScrapCancel = { - onClickScrapCancel() - onDismissCancelDialog() - } - ) + uiState.scrapId?.run { + ScrapCancelDialog( + scrapId = this, + onDismissRequest = onDismissCancelDialog + ) + } } if (uiState.internDialogVisibility) { - CalendarDetailDialog( - deadline = uiState.selectedDate.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.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 + ) + } } } 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..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 @@ -34,8 +21,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 +91,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 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..3e3c8fd77 --- /dev/null +++ b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelDialog.kt @@ -0,0 +1,125 @@ +package com.terning.feature.dialog.cancel + +import androidx.compose.foundation.layout.Column +import androidx.compose.foundation.layout.Spacer +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 +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 +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.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 +import com.terning.core.extension.toast +import com.terning.feature.R + +@Composable +fun ScrapCancelDialog( + scrapId: Long, + onDismissRequest: (Boolean) -> Unit, + viewModel: ScrapCancelViewModel = hiltViewModel() +) { + val context = LocalContext.current + 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 -> { + context.toast(sideEffect.message) + } + } + } + } + + ScrapCancelScreen( + onDismissRequest = { onDismissRequest(false) }, + onClickScrapCancel = { viewModel.deleteScrap(scrapId) } + ) + +} + +@Composable +private fun ScrapCancelScreen( + onDismissRequest: () -> Unit, + onClickScrapCancel: () -> Unit, +) { + TerningBasicDialog( + onDismissRequest = onDismissRequest + ) { + Column( + modifier = Modifier + .fillMaxWidth() + .wrapContentHeight() + .padding(top = 60.dp) + .padding(horizontal = 16.dp), + horizontalAlignment = Alignment.CenterHorizontally + ) { + TerningLottieAnimation( + jsonString = "terning_scrap_cancel.json", + modifier = Modifier + .fillMaxWidth() + .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 + ) + 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 + ) + Spacer(modifier = Modifier.height(16.dp)) + } + + } +} + + +@Preview(showBackground = true) +@Composable +private fun TerningBasicDialogPreview() { + TerningPointTheme { + ScrapCancelScreen( + onClickScrapCancel = {}, + 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..351729eaf --- /dev/null +++ b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelSideEffect.kt @@ -0,0 +1,8 @@ +package com.terning.feature.dialog.cancel + +import androidx.annotation.StringRes + +sealed class 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/cancel/ScrapCancelViewModel.kt b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelViewModel.kt new file mode 100644 index 000000000..0cf47d3a2 --- /dev/null +++ b/feature/src/main/java/com/terning/feature/dialog/cancel/ScrapCancelViewModel.kt @@ -0,0 +1,35 @@ +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 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 { + with(_sideEffect){ + emit(ScrapCancelSideEffect.DismissDialog) + emit(ScrapCancelSideEffect.ShowToast(R.string.dialog_scrap_cancelled)) + } + + }.onFailure { + _sideEffect.emit(ScrapCancelSideEffect.ShowToast(R.string.server_failure)) + } + } +} \ No newline at end of file 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..e160aea7a --- /dev/null +++ b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialog.kt @@ -0,0 +1,367 @@ +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.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.text.style.TextOverflow +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.Grey100 +import com.terning.core.designsystem.theme.Grey200 +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.extension.toast +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 + + +@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 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 -> { + context.toast(sideEffect.message) + } + is ScrapDialogSideEffect.DismissDialog -> { + viewModel.initUiState() + onDismissRequest() + } + + is ScrapDialogSideEffect.PatchedScrap -> onClickChangeColor() + is ScrapDialogSideEffect.NavigateToDetail -> onClickNavigateButton( + internshipAnnouncementId + ) + + is ScrapDialogSideEffect.ScrappedAnnouncement -> { + } + } + } + } + + LaunchedEffect(true) { + val colorType = ColorType.findColorType(scrapColor).takeIf { it != null } + colorType?.let { viewModel.setInitialAndSelectedColorType(it) } + } + + TerningBasicDialog( + onDismissRequest = viewModel::navigateUp + ) { + ScrapDialogScreen( + title = title, + deadline = deadline, + startYear = startYear, + startMonth = startMonth, + workingPeriod = workingPeriod, + isScrapped = isScrapped, + companyImage = companyImage, + selectedColorType = uiState.selectedColorType, + isColorChanged = uiState.isColorChanged, + isColorChangedOnce = uiState.isColorChangedOnce, + onClickColorButton = viewModel::changeSelectedColor, + onClickColorChangeButton = { + if(uiState.isColorChanged) + viewModel.patchScrap(scrapId = scrapId, color = uiState.selectedColorType) + }, + onClickNavigateButton = viewModel::navigateToDetail, + onClickScrapButton = { + viewModel.postScrap(internshipAnnouncementId, uiState.selectedColorType) + } + ) + } +} + + +@Composable +private fun ScrapDialogScreen( + title: String, + deadline: String, + startYear: Int, + startMonth: Int, + workingPeriod: String, + isScrapped: Boolean, + companyImage: String, + selectedColorType: ColorType, + isColorChanged: Boolean, + isColorChangedOnce: Boolean, + onClickColorButton: (ColorType) -> Unit, + onClickNavigateButton: () -> Unit, + onClickColorChangeButton: () -> Unit, + onClickScrapButton: () -> Unit +) { + 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) + ) + ) + + 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(10.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.body6, + color = Grey400, + ) + } + + Spacer(modifier = Modifier.height(8.dp)) + + Box( + modifier = Modifier.fillMaxWidth(), + contentAlignment = Alignment.Center + ) { + ColorPalette( + selectedColor = selectedColorType, + onColorSelected = onClickColorButton + ) + } + HorizontalDivider( + thickness = 1.dp, + color = Grey200, + modifier = Modifier.padding(vertical = 8.dp) + ) + + + Column( + modifier = Modifier.padding(vertical = 4.dp), + verticalArrangement = Arrangement.spacedBy( + 5.dp, + Alignment.CenterVertically + ), + horizontalAlignment = Alignment.Start, + ) { + InternInfoRow( + title = stringResource(id = R.string.intern_info_d_day), + value = deadline + ) + InternInfoRow( + title = stringResource(id = R.string.intern_info_working), + value = workingPeriod + ) + InternInfoRow( + title = stringResource(id = R.string.intern_info_start_date), + value = "${startYear}년 ${startMonth}월" + ) + } + } + + Spacer(modifier = Modifier.height(8.dp)) + + if (isScrapped) { + DetailScrapButton( + isColorChanged = isColorChanged, + isColorChangedOnce = isColorChangedOnce, + onClickNavigateButton = onClickNavigateButton, + onClickColorChangeButton = onClickColorChangeButton + + ) + } 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( + isColorChanged: Boolean, + isColorChangedOnce: Boolean, + onClickNavigateButton: () -> Unit, + onClickColorChangeButton: () -> Unit, + modifier: Modifier = Modifier +) { + Row( + modifier = modifier + ) { + ScrapColorChangeButton( + isColorChanged = isColorChanged, + isColorChangedOnce = isColorChangedOnce, + paddingVertical = 12.dp, + cornerRadius = 8.dp, + 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 = "터닝 하반기 채용터닝 하반기 채용터닝 하반기 채용터닝터닝 하반기 채용터닝터닝 하반기 채용터닝터닝 하반기 채용터닝터닝 하반기 채용터닝터닝 하반기 채용터닝 하반기 채용터닝 하반기 채용터닝 하반기 채용터닝 하반기 채용터닝 하반기 채용터닝 하반기 채용", + deadline = "2024/09/07", + startYear = 2024, + startMonth = 11, + workingPeriod = "2개월", + companyImage = "", + isScrapped = false, + onClickNavigateButton = {}, + onClickColorChangeButton = {}, + onClickScrapButton = {}, + onClickColorButton = {}, + isColorChanged = false, + isColorChangedOnce = true, + selectedColorType = ColorType.RED + ) + } +} + +@Preview(showBackground = true) +@Composable +private fun NewScrapButtonPreview() { + TerningPointTheme { + NewScrapButton( + onClickScrapButton = {} + ) + } +} + +@Preview(showBackground = true) +@Composable +private fun DetailScrapButtonPreview() { + TerningPointTheme { + DetailScrapButton( + onClickNavigateButton = {}, + onClickColorChangeButton = {}, + isColorChanged = false, + isColorChangedOnce = false + ) + } +} \ 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..67343309b --- /dev/null +++ b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogSideEffect.kt @@ -0,0 +1,11 @@ +package com.terning.feature.dialog.detail + +import androidx.annotation.StringRes + +sealed class ScrapDialogSideEffect{ + data class ShowToast(@StringRes val message: Int): ScrapDialogSideEffect() + data object DismissDialog : ScrapDialogSideEffect() + data object ScrappedAnnouncement : 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 new file mode 100644 index 000000000..3f08e2e4b --- /dev/null +++ b/feature/src/main/java/com/terning/feature/dialog/detail/ScrapDialogViewModel.kt @@ -0,0 +1,122 @@ +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.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 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 initUiState() { + _uiState.update { currentState -> + currentState.copy( + isColorChanged = false, + isColorChangedOnce = false + ) + } + } + + fun setInitialAndSelectedColorType(colorType: ColorType) { + _uiState.update { currentState -> + currentState.copy( + initialColorType = colorType, + selectedColorType = colorType + ) + } + } + + 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) { + val colorIndex = getColorIndex(color.main) + viewModelScope.launch { + scrapRepository.postScrap(CalendarScrapRequest(id, colorIndex)) + .onSuccess { + with(_sideEffect) { + emit(ScrapDialogSideEffect.ShowToast(R.string.dialog_scrap_scrapped)) + emit(ScrapDialogSideEffect.ScrappedAnnouncement) + } + }.onFailure { + _sideEffect.emit(ScrapDialogSideEffect.ShowToast(R.string.server_failure)) + } + } + } + + fun patchScrap(scrapId: Long, color: ColorType) = viewModelScope.launch { + val colorIndex = getColorIndex(color.main) + scrapRepository.patchScrap(CalendarScrapRequest(scrapId, colorIndex)) + .onSuccess { + _sideEffect.emit(ScrapDialogSideEffect.PatchedScrap) + _uiState.update { currentState -> + currentState.copy( + initialColorType = color, + isColorChanged = false, + isColorChangedOnce = true, + ) + } + }.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 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..f752e464c --- /dev/null +++ b/feature/src/main/java/com/terning/feature/dialog/detail/component/ColorPalette.kt @@ -0,0 +1,106 @@ +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.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( + selectedColor: ColorType, + onColorSelected: (ColorType) -> Unit, + modifier: Modifier = Modifier +) { + RadioButtonGroups( + options = ColorType.entries, + gridCellCount = ColorType.entries.size, + onOptionSelected = onColorSelected, + buttonComposable = { colorType, isSelected, onOptionSelected -> + ColorButton( + colorType = colorType, + 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 = "color selected" + ) + } + } + } +} + +@Preview(showBackground = true) +@Composable +fun ColorPalettePreview() { + ColorPalette( + 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..59fe57c74 --- /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.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 { + isColorChangedOnce && !isColorChanged -> TerningSub4 + isColorChanged && !isPressed -> White + isColorChanged && isPressed -> TerningSub5 + 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 d7500a8a3..e258ad031 100644 --- a/feature/src/main/res/values/strings.xml +++ b/feature/src/main/res/values/strings.xml @@ -54,6 +54,21 @@ 선택하신 날짜에 지원 마감인 스크랩 공고가 없어요 +%s + + 공고를 캘린더에 스크랩하시겠어요? + 스크랩 색상 + 색상 변경하기 + 색상 변경완료 + 내 캘린더에 스크랩하기 + 관심 공고가 캘린더에서 사라져요! + 스크랩을 취소하시겠어요? + 스크랩 취소하기 + 내가 스크랩한 관심 공고에요! + 공고 상세 정보 보기 + 오늘 마감되는 공고예요! + 관심 공고가 캘린더에 스크랩되었어요! + 관심 공고가 캘린더에서 사라졌어요! + 오늘 마감되는 %s님의 관심 공고 아직 스크랩된 인턴 공고가 없어요!\n관심 공고를 스크랩하면 마감 당일에 알려드릴게요