Project.swift 1.5 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152
  1. //
  2. // Project.swift
  3. // Todos
  4. //
  5. // Created by Sam Jaffe on 2/28/26.
  6. //
  7. import Foundation
  8. import SwiftData
  9. @Model
  10. final class Project: Codable, Ordered {
  11. var sortOrder: Int = 0
  12. var timestamp: Date
  13. var category: String = ""
  14. var name: String = "New Project"
  15. @Relationship(deleteRule: .cascade, inverse: \Task.project)
  16. var tasks: [Task] = []
  17. init(timestamp: Date) {
  18. self.timestamp = timestamp
  19. }
  20. func yaml(_ indent: Int = 0) -> String {
  21. let hdr = String(repeating: " ", count: indent)
  22. var rval = hdr + "\(name):\n"
  23. if !category.isEmpty {
  24. rval += hdr + " # In Category: \(category)\n"
  25. }
  26. rval += tasks.map({ $0.yaml(indent + 1) }).joined()
  27. return rval
  28. }
  29. enum CodingKeys: CodingKey { case timestamp, category, name, tasks }
  30. required init(from decoder: any Decoder) throws {
  31. let container = try decoder.container(keyedBy: CodingKeys.self)
  32. timestamp = try container.decode(Date.self, forKey: .timestamp)
  33. category = try container.decode(String.self, forKey: .category)
  34. name = try container.decode(String.self, forKey: .name)
  35. tasks = try container.decode([Task].self, forKey: .tasks)
  36. tasks.forEach({ $0.project = self })
  37. }
  38. func encode(to encoder: any Encoder) throws {
  39. var container = encoder.container(keyedBy: CodingKeys.self)
  40. try container.encode(timestamp, forKey: .timestamp)
  41. try container.encode(category, forKey: .category)
  42. try container.encode(tasks, forKey: .tasks)
  43. try container.encode(name, forKey: .name)
  44. }
  45. }