Skip to content

MaryJo-github/ios-diary

Repository files navigation


📖 목차

🍀 소개
💻 실행 화면
🛠️ 사용 기술
👀 다이어그램
🧨 트러블 슈팅
📚 참고 링크
👩‍👧‍👧 about TEAM


🍀 소개

📙 새 일기 작성 및 저장된 일기 수정이 가능한 일기장 앱입니다. 📗


💻 실행 화면

새 일기 작성 저장된 일기 수정

🛠️ 사용 기술

구현 내용 도구
아키텍쳐 MVC
UI UIKit
로깅방식 OS_Log
지원 언어 한국어, English

👀 Diagram

📐 UML


🧨 트러블 슈팅

1️⃣ DTO/Entity 분리

🚨 문제점

  • Json 샘플 파일에는 다이어리가 생성된 날짜를 timeInterval 형태(ex. 1608651333)로 가지고 있습니다. 하지만 ViewController에서는 생성 날짜를 원하는 포맷(ex. 2020년 12월 23일)으로 바꿔준 후 UI에 활용하기때문에, 활용할 때마다 DateFormat을 변경하는 것은 비효율적이라고 생각했습니다.

💡 해결방법

  • JSON decoding 시 활용하는 객체(DTO)와 실제 모델에서 활용하는 객체(Entity)를 분리해줌으로써 해결하였습니다.
    • 다이어리가 생성된 날짜를 timeInterval 형태로 가지는 DTO와 달리 Entity는 원하는 포맷으로 변환한 date를 가지고있습니다.
    • 다이어리를 생성하거나, decoding할 때에만 DateFormat을 변경하고, 이후에는 저장된 date를 이용합니다.

🔀 코드 변화

  • 수정 전
    struct DiaryContent: Codable {
        var title: String
        var body: String
        var timeInterval: Double
    
        private enum CodingKeys: String, CodingKey {
            case title, body
            case timeInterval = "created_at"
        }
    }
  • 수정 후
    // DTO
    struct DiaryContentDTO: Decodable {
        var title: String
        var body: String
        var timeInterval: Double
    
        private enum CodingKeys: String, CodingKey {
            case title, body
            case timeInterval = "created_at"
        }
    }
    
    // Entity
    struct DiaryContent {
        var title: String
        var body: String
        var date: String
    }

2️⃣ DiaryManager (STEP2 진행 후 수정 예정)

Asset에 넣은 "sample" 데이터를 받아오는 메서드를 구현하는데, 이 작업은 ViewController의 역할이 아니라고 생각해서 DiaryManager라는 객체를 Model로 만들었습니다. DiaryViewControllerDiaryManager를 가지고 있고, DiaryManager가 데이터 fetch 작업을 할 수 있도록 메서드를 구현하였습니다.

수정 전
struct DiaryManager {
    func fetchDiaryContents(name: String) -> [DiaryContent]? {
        do {
            let data: [DiaryContent] = try DecodingManager.decodeJSON(fileName: name)
            
            return data
        } catch {
            os_log("%{public}@", type: .default, error.localizedDescription)
        }
        
        return nil
    }
}
    
final class DiaryViewController: UIViewController {
    private let diaryManager = DiaryManager()
    private var data: [DiaryContent]?
            ...
    
    override func viewDidLoad() {
            ...
    
        fetchData()
    }
            ...
    
    private func fetchData() {
        data = diaryManager.fetchDiaryContents(name: "sample")
    }
}

🚨 문제점

  • DiaryViewControllerDiaryManager를 가지고 있긴 하지만 fetchData 메서드에서 manager의 메서드를 호출해 일을 시킴으로써 Controller가 데이터를 받아오는 일을 하게되고, 그로 인해 fetchDiaryContents 메서드를 은닉화 할 수 없었습니다.

💡 해결방법

  • DiaryViewController에서 필요한 diaryContentsDiaryManager의 연산프로퍼티로 구현하고, 그 연산에 fetchDiaryContents 메서드를 호출해서 문제점들을 해결하였습니다.
수정 후
struct DiaryManager {
    var diaryContents: [DiaryContent]? {
        return fetchDiaryContents(name: "sample")
    }
    
    private func fetchDiaryContents(name: String) -> [DiaryContent]? {
        do { ... } catch { ... }
    }
}
    
final class DiaryViewController: UIViewController {
    private let diaryManager = DiaryManager()
            ...
}
    
extension DiaryViewController: UITableViewDataSource {
    func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return diaryManager.diaryContents?.count ?? 0
    }
            ...
}

3️⃣ SwiftLint 활용

더 깔끔하고 통일성있는 컨벤션을 위해 SwiftLint를 적용했습니다.

🚨 문제점

  • SwiftLint를 프로젝트에 적용했더니 빌드 시 경고가 발생하였습니다.

💡 해결방법

  • SwiftLint에서 디폴트로 적용되는 Rule 때문에 발생한 문제였습니다.
  • .swiftlint 파일을 다음과 같이 수정하여 저희의 컨벤션과 맞춰줌으로써 해결하였습니다.
    disabled_rules:
        - trailing_whitespace
        - trailing_comma

📚 참고 링크



👩‍👧‍👧 about TEAM

🐬Whales🐬 https://github.com/WhalesJin
🐿️Mary🐿️ https://github.com/MaryJo-github

About

No description, website, or topics provided.

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Contributors 3

  •  
  •  
  •  

Languages