Skip to content

h-suo/ProjectManager

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 

Repository files navigation

프로젝트 매니저 (SwiftUI & TCA)

📖 목차

  1. 📢 소개
  2. 👤 팀원
  3. ⏱️ 타임라인
  4. 📊 UML & 파일트리
  5. 📱 실행 화면
  6. 🤔 고민한 부분
  7. 🔗 참고 링크

1. 📢 소개

프로젝트를 생성하고 관리하고 날짜에 맞게 진행하세요! 기한이 지난 프로젝트는 표시됩니다!

핵심 개념 및 경험

  • SwiftUI
    • SwiftUI를 이용하여 선언형 UI 구현
  • TCA
    • 프로젝트의 가독성 및 역할 분리를 위해 TCA 사용
  • SwiftData
    • 데이터를 로컬에 저장하기 위해 SwiftData를 이용한 저장 기능 구현
  • Firebase
    • 데이터를 리모트에 저장하기 위해 Firebase를 이용한 저장 기능 구현

2. 👤 팀원

Erick

3. ⏱️ 타임라인

SwiftUI 프로젝트 기간 : 2024.03.04 ~ 2024.04.03

날짜 내용
2023.03.04 ▫️ SwiftUI 리펙토링 파일 생성
2023.03.05 ▫️ TCA 패키지 추가 및 그룹 분리
▫️ Project 객체 생성
▫️ ProjectsFeature 생성
▫️ ProjectsView 생성 및 UI 구현
2023.03.07 ▫️ ProjectDetailFeature 생성
▫️ ProjectDetailView 생성 및 UI 구현
2023.03.08 ▫️ Project 업데이트 및 삭제 기능 구현
2023.03.09 ▫️ Projects 필터링 로직 추가
▫️ Project State 변경 기능 구현
2023.03.12 ▫️ 마감기한 초과 표시 기능 구현
2023.03.20 ▫️ Database Environment 생성
▫️ 로컬 저장 기능 구현
2023.03.29 ▫️ Firebase 패키지 추가
▫️ Database 추상화 및 Combine을 이용한 데이터 처리
2023.04.03 ▫️ FirebaseDatabase Environment 생성
▫️ firebaseDatabase의 작업을 merge를 이용해 병렬 처리

4. 📊 UML & 파일트리

UML


파일트리

ProjectManager
├── Application
│   └── ProjectManagerApp.swift
├── Feature
│   ├── Project.swift
│   ├── ProjectsFeature.swift
│   └── ProjectDetailFeature.swift
├── View
│   ├── ProjectsView.swift
│   ├── ProjectList.swift
│   ├── ProjectRow.swift
│   └── ProjectDetailView.swift
├── Environment
│   ├── Database
│   │   ├── DatabaseProtocol.swift
│   │   └── Database.swift
│   ├── SwiftData
│   │   ├── SwiftDataProject.swift
│   │   └── SwiftDatabase.swift
│   └── Firebase
│       ├── FirebaseProject.swift
│       └── FirebaseDatabase.swift
├── Util
│   ├── Extenstion
│   │   └── Calendar+.swift
│   └── UserReadableError.swift
└── Resource
    ├── Assets.xcassets
    └── GoogleService-Info.plist

5. 📱 실행 화면

프로젝트 생성
프로젝트 수정
프로젝트 이동
프로젝트 삭제

6. 🤔 고민한 부분

1️⃣ SwiftUI

SwiftUI로 UI를 구현하며, 사용자가 쉽게 사용하고 보기에 어색하지 않은 UI를 만들기 위해 고민했습니다.

Picker

Project에는 todo, doing, done 3가지 상태가 존재합니다. 사용자가 이러한 Project의 상태를 쉽게 변경할 수 있도록 모든 케이스와 선택된 케이스를 한 번에 보여주기 위해 Picker를 사용했습니다.

Picker는 서로 다른 데이터 모음에서 선택 제공하는 뷰로, Segmented 스타일의 Picker에 ProjectState의 모든 케이스를 Text로 넣어 사용자가 한 번에 보고 선택할 수 있도록 했습니다.

Picker("", selection: $store.project.projectState.sending(\.setProjectState)) {
  ForEach(ProjectState.allCases, id: \.self) { state in
    Text(state.rawValue)
  }
}
.pickerStyle(SegmentedPickerStyle())

overlay

TextField와 TextEditor가 함께 사용되는 뷰에서 TextField의 RoundedBorder 스타일과 TextEditor의 스타일을 통일하여 사용자에게 자연스러운 UI를 제공하기 위해 overlay를 활용했습니다.

overlay는 뷰의 앞에 특정 뷰를 레이어링 하기 위한 메서드로 TextEditor에 RoundedRectangle을 레이어링 하여 RoundedBorder 스타일과 동일한 UI를 구현했습니다.

TextEditor(text: $store.project.body.sending(\.setBody))
  .overlay {
    RoundedRectangle(cornerRadius: 8)
      .stroke(.placeholder, lineWidth: 0.5)
  }

2️⃣ TCA

SwiftUI의 View는 데이터 바인딩을 지원하기 때문에 MVVM의 View와 ViewModel의 구분이 모호해진다는 문제가 있었습니다. 하여 TCA를 사용해서 프로젝트를 설계했습니다.

TCA는 크게 View, Action, Reducer, State, Environment로 이루어져 있으며 하나의 View에 하나의 Store가 존재하며 Action을 통해 State를 변화시키는 단방향 플로우이기 때문에 흐름을 추적 관리하기 쉽다고 생각했습니다.

State

State는 UI를 그릴 때 필요한 데이터에 대한 설명을 나타내는 타입입니다.

Projects를 관리하는 ProjectsFeature에는 Projects로 UI를 그려야 하기 때문에 State에서 Projects를 가지고 있도록 했습니다.

struct State: Equatable {
  var projects: [Project]
}

Action

Action은 사용자가 하는 행동이나 노티피케이션 등 앱에서 생길 수 있는 모든 행동을 나타내는 타입입니다.

View의 이벤트나 사용자 이벤트에 대한 행동을 나타내고 처리하기 위해 onAppear, ButtonTapped, RowEvent 등을 지정했습니다.

enum Action {
  case onAppear
  case addButtonTapped
  case projectRowSelected(Project)
  case projectRowDeleted(Project)
  case fetchProjects(Result<[Project], DatabaseError>)
}

Reducer

Reducer는 Action이 주어졌을 때 Effect를 반환하거나 State를 변경시키는 방법을 가지고 있는 함수입니다.

onAppear Action이 주어졌을 때 publisher Effect를 반환합니다. publisher Effect는 publisher의 output을 파라미터로 받는 Action으로 변환하여 실행하는 Effect입니다.

var body: some Reducer<State, Action> {
  Reduce { state, action in
    switch action {
    case .onAppear:
      return .publisher {
        swiftDatabase.fetch()
          .merge(with: firebaseDatabase.fetch())
          .receive(on: DispatchQueue.main)
          .map(Action.fetchProjects)
      }
    case .addButtonTapped:
      // Button event handling
    case let .projectRowSelected(project):
      // Project Select handling
    case let .projectRowDeleted(project):
      // Project Delete handling
    case let .fetchProjects(result):
      switch result {
      case let .success(projects):
        state.projects = projects
      case let .failure(error):
        // Error handling
      }
      return .none
    }
  }

View

View에서는 State, Action, Reducer를 가지고 있는 Store를 가지고 UI를 그리거나 이벤트를 전달합니다.

struct ProjectsView: View {
  @Bindable private var store: StoreOf<ProjectsFeature>
  
  init(store: StoreOf<ProjectsFeature>) {
    self.store = store
  }
  
  var body: some View {
    // View
  }
}

3️⃣ SwiftData

로컬 DB를 간단히 구현하기 위해 SwiftData를 활용했습니다. 또한 TCA에 SwiftData를 적용하기 위해 고민했습니다.

Model

Project 객체에서 convert 할 수 있는 SwiftDataProject를 이용해 SwiftData에 데이터를 저장했습니다.

Attribute를 이용해 id가 모든 인스턴스에서 고유하도록 지정하여 Model 데이터의 충돌을 피할 수 있도록 했습니다.

@Model
final class SwiftDataProject {
  @Attribute(.unique) var id: UUID
  var title: String
  var body: String
  var deadline: Date
  var projectState: ProjectState
  
  init(
    id: UUID,
    title: String,
    body: String,
    deadline: Date,
    projectState: ProjectState
  ) {
    self.id = id
    self.title = title
    self.body = body
    self.deadline = deadline
    self.projectState = projectState
  }
}

DependencyKey

SwiftData를 TCA에서 쉽게 사용할 수 있도록 DependencyKey를 이용해서 의존성 관리를 했습니다.

SwiftDatabase를 생성하여 Project를 검색, 저장, 삭제하는 객체의 인터페이스를 지정했습니다.

struct SwiftDatabase: DatabaseProtocol {
  var fetch: () -> AnyPublisher<Result<[Project], DatabaseError>, Never>
  var add: (Project) -> AnyPublisher<Result<Project, DatabaseError>, Never>
  var delete: (Project) -> AnyPublisher<Result<Project, DatabaseError>, Never>
}

의존성 관리를 위해 DependencyKey를 채택하고 인터페이스의 동작을 정의했습니다.

ModelContext를 이용해 Project의 검색, 저장, 삭제를 구현했습니다.

extension SwiftDatabase: DependencyKey {
  static var liveValue: SwiftDatabase = Self(
    fetch: {
      do {
        @Dependency(\.database.modelContext) var context
        let projectContext = try context()
        let descriptor = FetchDescriptor<SwiftDataProject>(sortBy: [SortDescriptor(\.deadline)])
        let projects = try projectContext.fetch(descriptor).map { $0.convertToProject() }
        return Just(.success(projects))
          .eraseToAnyPublisher()
      } catch {
        return Just(.failure(.fetchFailed(error)))
          .eraseToAnyPublisher()
      }
    },
    add: { project in
      // Add Project handling
    },
    delete: { project in
      // Delete Project handling
    }
  )
}

SwiftDatabase를 DependencyValues에 등록하여 Reducer에서 쉽게 접근할 수 있도록 했습니다.

extension DependencyValues {
  var swiftDatabase: SwiftDatabase {
    get { self[SwiftDatabase.self] }
    set { self[SwiftDatabase.self] = newValue }
  }
}

4️⃣ Firebase

리모트 DB를 구현하기 위해 Firebase의 Firestore를 사용했습니다.

Codable

Project 객체에서 convert 할 수 있는 FirebaseProject를 이용해 Firestore에 데이터를 저장했습니다.

FirebaseProject는 Codable을 따르도록 하여 데이터 송수신 시 쉽게 변환할 수 있도록 했습니다.

struct FirebaseProject: Codable {
  var id: String
  var title: String
  var body: String
  var deadline: Date
  var projectState: ProjectState
}

extension FirebaseProject {
  func convertToProject() -> Project {
    Project(
      id: UUID(uuidString: id)!,
      title: title,
      body: body,
      deadline: deadline,
      projectState: projectState
    )
  }
}

DependencyKey

SwiftData와 같이 Firebase를 TCA에서 쉽게 사용할 수 있도록 DependencyKey를 이용해서 의존성 관리를 했습니다.

Merge

ProjectsFeature에서 데이터 검색, 저장, 삭제의 작업을 SwiftData와 Firebase가 모두 수행해야 했습니다.

SwiftData와 Firebase를 두 개의 스트림으로 처리하지 않고, 하나의 스트림으로 처리하기 위해 merge를 이용해 작업을 처리했습니다.

return .publisher {
  swiftDatabase.fetch()
    .merge(with: firebaseDatabase.fetch())
    .receive(on: DispatchQueue.main)
    .map(Action.fetchProjects)
}

7. 🔗 참고 링크



⬆️ 처음으로 돌아가기

About

프로젝트 관리 앱 저장소

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages