Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

출석 뷰 컴포즈로 마이그레이션 (진행중) #990

Draft
wants to merge 7 commits into
base: develop
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions app/build.gradle.kts
Original file line number Diff line number Diff line change
Expand Up @@ -148,6 +148,7 @@ dependencies {
implementation(libs.kotlin.coroutines.google.play)
implementation(platform(libs.compose.bom))
implementation(libs.bundles.compose)
implementation(libs.compose.lifecycle)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

어라 이거 bundles.compose에 없었나..?

implementation(libs.startup)
implementation(libs.swipe.refresh.layout)
debugImplementation(libs.compose.ui.tooling)
Expand Down Expand Up @@ -177,6 +178,7 @@ dependencies {
implementation(libs.coil.core)
implementation(libs.profileinstaller)
implementation(libs.firebase.messaging.lifecycle.ktx)
implementation(libs.kotlin.collections.immutable)
}

secrets {
Expand Down
5 changes: 4 additions & 1 deletion app/src/main/AndroidManifest.xml
Original file line number Diff line number Diff line change
Expand Up @@ -153,5 +153,8 @@
<action android:name="com.google.firebase.MESSAGING_EVENT" />
</intent-filter>
</service>
<activity
android:name=".feature.attendance.NewAttendanceActivity"
android:exported="true" />
</application>
</manifest>
</manifest>
84 changes: 84 additions & 0 deletions app/src/main/java/org/sopt/official/data/AttendanceMapper.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,84 @@
package org.sopt.official.data

import org.sopt.official.data.model.attendance.AttendanceHistoryResponse
import org.sopt.official.data.model.attendance.AttendanceHistoryResponse.AttendanceResponse
import org.sopt.official.data.model.attendance.SoptEventResponse
import org.sopt.official.domain.entity.attendance.Attendance
import org.sopt.official.domain.entity.attendance.Attendance.AttendanceDayType
import org.sopt.official.domain.entity.attendance.Attendance.AttendanceDayType.HasAttendance.RoundAttendance
import org.sopt.official.domain.entity.attendance.Attendance.AttendanceDayType.HasAttendance.RoundAttendance.RoundAttendanceState
import org.sopt.official.domain.entity.attendance.Attendance.Session
import org.sopt.official.domain.entity.attendance.Attendance.User.AttendanceLog.AttendanceState
import java.time.LocalDateTime

fun mapToAttendance(
attendanceHistoryResponse: AttendanceHistoryResponse?,
soptEventResponse: SoptEventResponse?
): Attendance {
return Attendance(
sessionId = soptEventResponse?.id ?: Attendance.UNKNOWN_SESSION_ID,
user = Attendance.User(
name = attendanceHistoryResponse?.name ?: Attendance.User.UNKNOWN_NAME,
generation = attendanceHistoryResponse?.generation ?: Attendance.User.UNKNOWN_GENERATION,
part = Attendance.User.Part.valueOf(attendanceHistoryResponse?.part ?: Attendance.User.UNKNOWN_PART),
attendanceScore = attendanceHistoryResponse?.score ?: 0.0,
attendanceCount = Attendance.User.AttendanceCount(
attendanceCount = attendanceHistoryResponse?.attendanceCount?.normal ?: 0,
lateCount = attendanceHistoryResponse?.attendanceCount?.late ?: 0,
absenceCount = attendanceHistoryResponse?.attendanceCount?.abnormal ?: 0,
),
attendanceHistory = attendanceHistoryResponse?.attendances?.map { attendanceResponse: AttendanceResponse ->
Attendance.User.AttendanceLog(
sessionName = attendanceResponse.eventName,
date = attendanceResponse.date,
attendanceState = AttendanceState.valueOf(attendanceResponse.attendanceState)
)
} ?: emptyList(),
),
attendanceDayType = soptEventResponse.toAttendanceDayType()
)
}

private fun SoptEventResponse?.toAttendanceDayType(): AttendanceDayType {
return when (this?.type) {
"HAS_ATTENDANCE" -> {
val firstAttendanceResponse: SoptEventResponse.AttendanceResponse? = attendances.getOrNull(0)
val secondAttendanceResponse: SoptEventResponse.AttendanceResponse? = attendances.getOrNull(1)
AttendanceDayType.HasAttendance(
session = Session(
name = eventName,
location = location.ifBlank { null },
startAt = LocalDateTime.parse(startAt),
endAt = LocalDateTime.parse(endAt),
),
firstRoundAttendance = RoundAttendance(
state = RoundAttendanceState.valueOf(firstAttendanceResponse?.status ?: RoundAttendanceState.NOT_YET.name),
attendedAt = LocalDateTime.parse(firstAttendanceResponse?.attendedAt),
),
secondRoundAttendance = RoundAttendance(
state = RoundAttendanceState.valueOf(secondAttendanceResponse?.status ?: RoundAttendanceState.NOT_YET.name),
attendedAt = LocalDateTime.parse(secondAttendanceResponse?.attendedAt),
),
)
}

"NO_ATTENDANCE" -> {
AttendanceDayType.NoAttendance(
session = Session(
name = eventName,
location = location.ifBlank { null },
startAt = LocalDateTime.parse(startAt),
endAt = LocalDateTime.parse(endAt),
)
)
}

"NO_SESSION" -> {
AttendanceDayType.NoSession
}

else -> {
AttendanceDayType.NoSession
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,73 @@
package org.sopt.official.data.repository.attendance

import kotlinx.serialization.json.Json
import kotlinx.serialization.json.JsonObject
import kotlinx.serialization.json.contentOrNull
import kotlinx.serialization.json.jsonObject
import kotlinx.serialization.json.jsonPrimitive
import org.sopt.official.data.mapToAttendance
import org.sopt.official.data.model.attendance.AttendanceHistoryResponse
import org.sopt.official.data.model.attendance.AttendanceRoundResponse
import org.sopt.official.data.model.attendance.RequestAttendanceCode
import org.sopt.official.data.model.attendance.SoptEventResponse
import org.sopt.official.data.service.attendance.AttendanceService
import org.sopt.official.domain.entity.attendance.Attendance
import org.sopt.official.domain.entity.attendance.ConfirmAttendanceCodeResult
import org.sopt.official.domain.entity.attendance.FetchAttendanceCurrentRoundResult
import org.sopt.official.domain.repository.attendance.NewAttendanceRepository
import retrofit2.HttpException
import javax.inject.Inject

class DefaultAttendanceRepository @Inject constructor(
private val attendanceService: AttendanceService,
private val json: Json
) : NewAttendanceRepository {
override suspend fun fetchAttendanceInfo(): Attendance {
val soptEventResponse: SoptEventResponse? = runCatching { attendanceService.getSoptEvent().data }.getOrNull()
val attendanceHistoryResponse: AttendanceHistoryResponse? =
runCatching { attendanceService.getAttendanceHistory().data }.getOrNull()

val attendance: Attendance =
mapToAttendance(attendanceHistoryResponse = attendanceHistoryResponse, soptEventResponse = soptEventResponse)
return attendance
}

override suspend fun fetchAttendanceCurrentRound(lectureId: Long): FetchAttendanceCurrentRoundResult {
return runCatching { attendanceService.getAttendanceRound(lectureId).data }.fold(
onSuccess = { attendanceRoundResponse: AttendanceRoundResponse? ->
FetchAttendanceCurrentRoundResult.Success(attendanceRoundResponse?.round)
},
onFailure = { error: Throwable ->
if (error !is HttpException) return FetchAttendanceCurrentRoundResult.Failure(null)

val message: String? = error.jsonErrorMessage
FetchAttendanceCurrentRoundResult.Failure(message)
},
)
}

override suspend fun confirmAttendanceCode(
subLectureId: Long,
code: String
): ConfirmAttendanceCodeResult {
return runCatching {
attendanceService.confirmAttendanceCode(RequestAttendanceCode(subLectureId = subLectureId, code = code))
}.fold(
onSuccess = { ConfirmAttendanceCodeResult.Success },
onFailure = { error: Throwable ->
if (error !is HttpException) return ConfirmAttendanceCodeResult.Failure(null)

val message: String? = error.jsonErrorMessage
ConfirmAttendanceCodeResult.Failure(message)
},
)
}

private val HttpException.jsonErrorMessage: String?
get() {
val errorBody: String = this.response()?.errorBody()?.string() ?: return null
val jsonObject: JsonObject = json.parseToJsonElement(errorBody).jsonObject
val errorMessage: String? = jsonObject["message"]?.jsonPrimitive?.contentOrNull
return errorMessage
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -29,12 +29,14 @@ import dagger.Module
import dagger.Provides
import dagger.hilt.InstallIn
import dagger.hilt.components.SingletonComponent
import javax.inject.Singleton
import org.sopt.official.common.di.OperationRetrofit
import org.sopt.official.data.repository.attendance.AttendanceRepositoryImpl
import org.sopt.official.data.repository.attendance.DefaultAttendanceRepository
import org.sopt.official.data.service.attendance.AttendanceService
import org.sopt.official.domain.repository.attendance.AttendanceRepository
import org.sopt.official.domain.repository.attendance.NewAttendanceRepository
import retrofit2.Retrofit
import javax.inject.Singleton

@Module
@InstallIn(SingletonComponent::class)
Expand All @@ -43,6 +45,10 @@ abstract class AttendanceBindsModule {
@Singleton
abstract fun bindAttendanceRepository(attendanceRepositoryImpl: AttendanceRepositoryImpl): AttendanceRepository

@Binds
@Singleton
abstract fun bindDefaultAttendanceRepository(defaultAttendanceRepository: DefaultAttendanceRepository): NewAttendanceRepository

companion object {
@Provides
@Singleton
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,111 @@
package org.sopt.official.domain.entity.attendance

import java.time.LocalDateTime

data class Attendance(
val sessionId: Int,
val user: User,
val attendanceDayType: AttendanceDayType,
) {
data class User(
val name: String,
val generation: Int,
val part: Part,
val attendanceScore: Number,
val attendanceCount: AttendanceCount,
val attendanceHistory: List<AttendanceLog>
) {
enum class Part(val partName: String) {
PLAN("기획"),
DESIGN("디자인"),
ANDROID("안드로이드"),
IOS("iOS"),
WEB("웹"),
SERVER("서버"),
UNKNOWN("")
}

data class AttendanceCount(
/** 출석 전체 횟수 */
val attendanceCount: Int,
/** 지각 전체 횟수 */
val lateCount: Int,
/** 결석 전체 횟수 */
val absenceCount: Int,
) {
/** 전체 횟수 */
val totalCount: Int
get() = attendanceCount + lateCount + absenceCount
}

data class AttendanceLog(
val sessionName: String,
val date: String,
val attendanceState: AttendanceState
) {
enum class AttendanceState {
/** 참여(출석 체크 X)*/
PARTICIPATE,

/** 출석 */
ATTENDANCE,

/** 지각 */
TARDY,

/** 결석 */
ABSENT
}
}

companion object {
const val UNKNOWN_NAME = "회원"
const val UNKNOWN_GENERATION = -1
const val UNKNOWN_PART = "UNKNOWN"
}
}

sealed interface AttendanceDayType {

/** 일정이 없는 날 */
data object NoSession : AttendanceDayType

/** 일정이 있고, 출석 체크가 있는 날 */
data class HasAttendance(
val session: Session,
val firstRoundAttendance: RoundAttendance,
val secondRoundAttendance: RoundAttendance
) : AttendanceDayType {
/** n차 출석에 관한 정보 */
data class RoundAttendance(
val state: RoundAttendanceState,
val attendedAt: LocalDateTime?
) {
/** n차 출석 상태 */
enum class RoundAttendanceState {
ABSENT, ATTENDANCE, NOT_YET,
}
}
}

/** 일정이 있고, 출석 체크가 없는 날 */
data class NoAttendance(val session: Session) : AttendanceDayType
}

/** 솝트의 세션에 관한 정보
* @property name 세션 이름 (OT, 1차 세미나, 솝커톤 등)
* @property location 세션 장소, 정해진 장소가 없을 경우(온라인) null
* @property startAt 세션 시작 시각
* @property endAt 세션 종료 시각
* */
data class Session(
val name: String,
val location: String?,
val startAt: LocalDateTime,
val endAt: LocalDateTime,
)

companion object {
const val UNKNOWN_SESSION_ID = -1
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
package org.sopt.official.domain.entity.attendance

sealed interface ConfirmAttendanceCodeResult {
data object Success : ConfirmAttendanceCodeResult
data class Failure(val errorMessage: String?) : ConfirmAttendanceCodeResult
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
package org.sopt.official.domain.entity.attendance

sealed interface FetchAttendanceCurrentRoundResult {
data class Success(val round: Int?) : FetchAttendanceCurrentRoundResult
data class Failure(val errorMessage: String?) : FetchAttendanceCurrentRoundResult
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
package org.sopt.official.domain.repository.attendance

import org.sopt.official.domain.entity.attendance.Attendance
import org.sopt.official.domain.entity.attendance.ConfirmAttendanceCodeResult
import org.sopt.official.domain.entity.attendance.FetchAttendanceCurrentRoundResult

interface NewAttendanceRepository {
suspend fun fetchAttendanceInfo(): Attendance
suspend fun fetchAttendanceCurrentRound(lectureId: Long): FetchAttendanceCurrentRoundResult
suspend fun confirmAttendanceCode(subLectureId: Long, code: String): ConfirmAttendanceCodeResult
}
Loading
Loading