- 코드의 Architecture는 Clean Architecture를 기반으로 한다.
- Multi Module 구조로 구성한다.
- Module의 구조는 각 Repository의 Readme에서 확인할 수 있다.
- MVVM 패턴을 사용한다.
- SAA 패턴을 사용한다.
- ViewModel(Presentation Layer)에는 안드로이드 프레임워크 관련 코드가 없어야함
- import android.* 코드가 없도록 유지(AAC관련 코드 제외, android.arch.*)
- https://medium.com/androiddevelopers/viewmodels-and-livedata-patterns-antipatterns-21efaef74a54
- 다만 데이터에 해당되는
R.string.xxx
,R.dimen.xxx
,R.color.xxx
들은 사용할 수 있도록 함- (
R.id.xxx
,R.layout.xxx
는 불가)
- (
SavedStateHandle
은 생성자의 첫번째에 위치한다.
Do
@HiltViewModel
class XxxViewModel @Inject constructor(
savedStateHandle: SavedStateHandle,
private val getFooUseCase: GetFooUseCase,
) : ViewModel()
Do not
@HiltViewModel
class XxxViewModel @Inject constructor(
private val getFooUseCase: GetFooUseCase,
savedStateHandle: SavedStateHandle,
private val getBarUseCase: GetBarUseCase,
) : ViewModel()
@HiltViewModel
class XxxViewModel @Inject constructor(
private val getFooUseCase: GetFooUseCase,
private val getBarUseCase: GetBarUseCase,
savedStateHandle: SavedStateHandle,
) : ViewModel()
: MVVM의 ViewModel에서 이벤트를 처리하는 방법 6가지
이벤트 이름은 {행위}-{대상}-{결과} 순서의 이름 규칙을 가진다.
object ShowCarInfo: Event() // 행위: 보여준다, 대상: 차량 정보
object UploadImageSuccess: Event() // 행위: 업로드하다. 대상: 이미지, 결과: 성공
- Event의 의미에 따라 동사의 형태는 자유롭게 사용한다. (과거형, 현재형)
- Success, Complete와 같이 의미가 이중적일 때에는 기대하는 동작에 따라 적절한 이름을 선택한다.
- 클래스는 각 한개의 usecase를 갖는다.
- 클래스명은 xxxUseCase 와 같이 명명한다.
- List를 가져오는 UseCase의 이름은 GetXXXListUseCase와 같이 정의한다.
- 함수의 경우 getXXXs()로 정의되나 단수를 가져오는 UseCase와 혼동의 우려가 있기때문에 UseCase이름은 List를 붙인다.
- 만약 Room DB를 사용하는 domain 클래스에 NonNull인 필드를 새로 추가하게 된다면 Local Class에서는 어쩔수없이 Nullable로 선언하고 mapping할때 default value를 지정하는 방법으로 구현해야 함
- 예)
- User에 email이 추가되어야함
- email이 NonNull이기때문에 UserLocal에 필드를 추가할때 email을 NonNull로 선언해야 할것 같지만
- 기존에 DB에 저장되어 있는 데이터에는 email 값이 없으므로 기존에 저장된 db를 toData()로 하려고 할때 문제가 발생함
- XXXLocal의 class는 새로 추가될때 항상 Nullable로 만들고 toData()로 mapping될때 기본값을 선언해줘야 함
- Model은 각각의 layer에 만들어져서 mapping되도록 구성한다
- 예) User라는 개념의 model에 대한 각각 layer에서의 정의
data class UserResponseDto(){}
fun UserResponseDto.toDomain(): UserEntity
data class UserEntity(){}
fun User.toData(): UserResponseDto
fun UserEntity.toPresentation(): UserModel
data class UserModel(){}
fun UserModel.toDomain(): UserEntity