Skip to content

Commit

Permalink
[MERGE] #9 -> develop
Browse files Browse the repository at this point in the history
[UI/#9] ์›”๊ฐ„ ์บ˜๋ฆฐ๋” ํ™”๋ฉด
  • Loading branch information
boiledEgg-s authored Jul 9, 2024
2 parents 8302624 + a86695a commit 95ce7d0
Show file tree
Hide file tree
Showing 23 changed files with 920 additions and 23 deletions.
6 changes: 6 additions & 0 deletions core/src/main/java/com/terning/core/extension/LocalDateExt.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
package com.terning.core.extension

import java.time.LocalDate

fun LocalDate.getStringAsTitle(): String =
"${year}๋…„ ${monthValue.toString().padStart(2, '0')}์›”"
4 changes: 4 additions & 0 deletions feature/build.gradle.kts
Original file line number Diff line number Diff line change
Expand Up @@ -91,6 +91,10 @@ dependencies {
implementation(libs.ossLicense)
implementation(libs.lottie)

//Compose Preview
implementation(libs.androidx.compose.ui.tooling)
implementation(libs.androidx.compose.ui.tooling.preview)

// KakaoDependencies
implementation(libs.kakao.user)

Expand Down
103 changes: 103 additions & 0 deletions feature/src/main/java/com/terning/feature/calendar/CalendarDay.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,103 @@
package com.terning.feature.calendar

import androidx.compose.foundation.background
import androidx.compose.foundation.clickable
import androidx.compose.foundation.interaction.MutableInteractionSource
import androidx.compose.foundation.layout.Box
import androidx.compose.foundation.layout.Row
import androidx.compose.foundation.layout.size
import androidx.compose.foundation.shape.CircleShape
import androidx.compose.material3.Text
import androidx.compose.runtime.Composable
import androidx.compose.runtime.remember
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
import androidx.compose.ui.graphics.Color
import androidx.compose.ui.tooling.preview.Preview
import androidx.compose.ui.unit.dp
import com.terning.core.designsystem.theme.Black
import com.terning.core.designsystem.theme.Grey150
import com.terning.core.designsystem.theme.Grey200
import com.terning.core.designsystem.theme.TerningMain
import com.terning.core.designsystem.theme.TerningTheme
import com.terning.core.designsystem.theme.White
import com.terning.feature.calendar.models.DayModel
import java.time.LocalDate

@Composable
fun CalendarDay(
modifier: Modifier = Modifier,
dayData: DayModel,
isSelected: Boolean,
isToday: Boolean,
onDateSelected: (LocalDate) -> Unit
) {
val backgroundColor =
if (isSelected) TerningMain else if (isToday) Grey200 else Color.Transparent
val textColor =
if (dayData.isOutDate) {
Grey150
} else {
if (isSelected)
White
else
Black
}

Box(
modifier = modifier,
contentAlignment = Alignment.Center
) {
Box(
modifier = Modifier
.size(22.dp)
.clickable(
interactionSource = remember { MutableInteractionSource() },
indication = null,
onClick = {
if (!dayData.isOutDate) {
onDateSelected(dayData.date)
}
}
)
.background(
color = backgroundColor,
shape = CircleShape
),
contentAlignment = Alignment.Center
) {
Text(
text = dayData.date.dayOfMonth.toString(),
color = textColor,
style = TerningTheme.typography.calendar
)
}
}
}

@Preview(showBackground = true)
@Composable
fun CalendarDayPreview() {
TerningTheme {
Row {
CalendarDay(
dayData = DayModel(LocalDate.now(), false),
isSelected = true,
isToday = true,
onDateSelected = {}
)
CalendarDay(
dayData = DayModel(LocalDate.now(), false),
isSelected = false,
isToday = true,
onDateSelected = {}
)
CalendarDay(
dayData = DayModel(LocalDate.now(), false),
isSelected = false,
isToday = false,
onDateSelected = {}
)
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,86 @@
package com.terning.feature.calendar

import androidx.compose.foundation.background
import androidx.compose.foundation.layout.Column
import androidx.compose.foundation.layout.Row
import androidx.compose.foundation.layout.Spacer
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.runtime.Composable
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
import androidx.compose.ui.tooling.preview.Preview
import androidx.compose.ui.unit.dp
import com.terning.core.designsystem.theme.Grey150
import com.terning.core.designsystem.theme.TerningTheme
import com.terning.feature.calendar.models.MonthData
import com.terning.feature.calendar.models.Scrap
import java.time.LocalDate
import java.time.YearMonth

@Composable
fun CalendarMonth(
modifier: Modifier = Modifier,
monthData: MonthData,
onDateSelected: (LocalDate) -> Unit,
selectedDate: LocalDate,
scrapLists: List<List<Scrap>> = listOf()
) {
Column(
modifier = modifier
.fillMaxSize()
.padding(horizontal = 20.dp),
) {
val month = monthData.calendarMonth.weekDays
for (week in month) {
Row(
modifier = Modifier.weight(1f),
) {
for (day in week) {
Column(
modifier = Modifier
.weight(1f)
.padding(top = 15.dp),
horizontalAlignment = Alignment.CenterHorizontally
) {
CalendarDay(
dayData = day,
isSelected = selectedDate == day.date,
isToday = day.date == LocalDate.now(),
onDateSelected = onDateSelected
)
if(!day.isOutDate) {
val index = day.date.dayOfWeek.value - 1
CalendarScrap(
scrapList = scrapLists[index]
)
}
}
}
}
if(month.indexOf(week) != month.lastIndex) {
Spacer(
modifier = Modifier
.fillMaxWidth()
.height(1.dp)
.background(color = Grey150)
)
}
}
}
}

@Preview(showBackground = true)
@Composable
fun CalendarMonthPreview() {
TerningTheme {
CalendarMonth(
monthData = MonthData(YearMonth.now()),
selectedDate = LocalDate.now(),
onDateSelected = {},
scrapLists = listOf()
)
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
package com.terning.feature.calendar

import androidx.compose.foundation.background
import androidx.compose.foundation.layout.fillMaxSize
import androidx.compose.foundation.lazy.LazyListState
import androidx.compose.foundation.lazy.LazyRow
import androidx.compose.runtime.Composable
import androidx.compose.ui.Modifier
import com.terning.core.designsystem.theme.White
import com.terning.feature.calendar.models.CalendarDefaults.flingBehavior
import com.terning.feature.calendar.models.CalendarState.Companion.getDateByPage
import com.terning.feature.calendar.models.MonthData
import com.terning.feature.calendar.models.Scrap
import java.time.LocalDate
import java.time.YearMonth

@Composable
fun CalendarMonths(
modifier: Modifier = Modifier,
listState: LazyListState,
onDateSelected: (LocalDate) -> Unit,
pages: Int,
selectedDate: LocalDate,
scrapLists: List<List<Scrap>>
) {
LazyRow(
modifier = modifier
.background(White),
state = listState,
userScrollEnabled = true,
flingBehavior = flingBehavior(
state = listState
)
) {
items(pages) { page ->
val date = getDateByPage(page)
val monthData = MonthData(YearMonth.of(date.year, date.month))

CalendarMonth(
modifier = Modifier.fillParentMaxSize(),
selectedDate = selectedDate,
onDateSelected = onDateSelected,
monthData = monthData,
scrapLists = scrapLists
)
}
}
}
116 changes: 116 additions & 0 deletions feature/src/main/java/com/terning/feature/calendar/CalendarRoute.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,116 @@
package com.terning.feature.calendar

import androidx.activity.compose.BackHandler
import androidx.compose.foundation.background
import androidx.compose.foundation.layout.Column
import androidx.compose.foundation.layout.Spacer
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.lazy.rememberLazyListState
import androidx.compose.material3.Scaffold
import androidx.compose.runtime.Composable
import androidx.compose.runtime.LaunchedEffect
import androidx.compose.runtime.collectAsState
import androidx.compose.runtime.getValue
import androidx.compose.runtime.mutableIntStateOf
import androidx.compose.runtime.mutableStateOf
import androidx.compose.runtime.remember
import androidx.compose.runtime.rememberCoroutineScope
import androidx.compose.runtime.setValue
import androidx.compose.runtime.snapshotFlow
import androidx.compose.ui.Modifier
import androidx.compose.ui.unit.dp
import androidx.hilt.navigation.compose.hiltViewModel
import com.terning.core.designsystem.theme.Grey200
import com.terning.feature.calendar.component.CalendarTopBar
import com.terning.feature.calendar.component.WeekDaysHeader
import com.terning.feature.calendar.models.CalendarState
import kotlinx.coroutines.flow.distinctUntilChanged
import kotlinx.coroutines.launch

@Composable
fun CalendarRoute() {
CalendarScreen()
}

@Composable
fun CalendarScreen(
modifier: Modifier = Modifier,
viewModel: CalendarViewModel = hiltViewModel()
) {
val selectedDate by viewModel.selectedDate.collectAsState()
val state by remember{ mutableStateOf(CalendarState()) }

val listState = rememberLazyListState(
initialFirstVisibleItemIndex = state.getInitialPage()
)

var currentDate by remember { mutableStateOf(selectedDate) }
var currentPage by remember { mutableIntStateOf(listState.firstVisibleItemIndex)}

var isListExpanded by remember { mutableStateOf(false) }
var isWeekEnabled by remember { mutableStateOf(false) }

LaunchedEffect(key1 = listState) {
snapshotFlow { listState.firstVisibleItemIndex }
.distinctUntilChanged()
.collect{
val swipeDirection = (listState.firstVisibleItemIndex - currentPage).coerceIn(-1, 1).toLong()
currentDate = currentDate.plusMonths(swipeDirection)
currentPage = listState.firstVisibleItemIndex
}
}

LaunchedEffect(key1 = selectedDate) {
isWeekEnabled = true
}

BackHandler {
isWeekEnabled = false
}

Scaffold(
modifier = modifier,
topBar = {
val coroutineScope = rememberCoroutineScope()
CalendarTopBar(
date = currentDate,
isListExpanded = isListExpanded,
onListButtonClicked = { isListExpanded = !isListExpanded },
onMonthNavigationButtonClicked = { direction ->
coroutineScope.launch {
listState.animateScrollToItem(
index = listState.firstVisibleItemIndex + direction,
)
}
}
)
}
) { paddingValues ->
Column(
modifier = Modifier
.fillMaxWidth()
.padding(top = paddingValues.calculateTopPadding())
) {
WeekDaysHeader()
Spacer(modifier = Modifier
.fillMaxWidth()
.height(1.dp)
.background(color = Grey200)
)
CalendarMonths(
modifier = Modifier.fillMaxSize(),
selectedDate = selectedDate,
onDateSelected = {
viewModel.updateSelectedDate(it)
},
listState = listState,
pages = state.getPageCount(),
scrapLists = viewModel.mockScrapList
)
}

}
}

This file was deleted.

Loading

0 comments on commit 95ce7d0

Please sign in to comment.