Project.swift 2.0 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273
  1. //
  2. // Project.swift
  3. // Todos
  4. //
  5. // Created by Sam Jaffe on 2/28/26.
  6. //
  7. import Foundation
  8. import SwiftData
  9. import SwiftUI
  10. @Model
  11. final class Project: Codable, Ordered, Aggregate {
  12. typealias Element = Task
  13. var uuid: UUID = UUID()
  14. var sortOrder: Int = 0
  15. var timestamp: Date
  16. var name: String = "New Project"
  17. var category: String = ""
  18. @Relationship(deleteRule: .cascade, inverse: \Task.project)
  19. var tasks: [Task] = []
  20. init(timestamp: Date, sortOrder: Int = 0) {
  21. self.timestamp = timestamp
  22. self.sortOrder = sortOrder
  23. }
  24. func move(fromOffsets: IndexSet, toOffset: Int) {
  25. tasks.move(fromOffsets: fromOffsets, toOffset: toOffset)
  26. reindex()
  27. }
  28. func remove(_ item: Element) {
  29. tasks.removeAll(where: { $0.id == item.id })
  30. reindex()
  31. }
  32. func reindex() {
  33. for (index, item) in tasks.enumerated() {
  34. item.sortOrder = index
  35. }
  36. }
  37. func yaml(_ indent: Int = 0) -> String {
  38. let hdr = String(repeating: " ", count: indent)
  39. var rval = hdr + "\(name):\n"
  40. if !category.isEmpty {
  41. rval += hdr + " # In Category: \(category)\n"
  42. }
  43. rval += tasks.map({ $0.yaml(indent + 1) }).joined()
  44. return rval
  45. }
  46. enum CodingKeys: CodingKey { case timestamp, category, name, tasks }
  47. required init(from decoder: any Decoder) throws {
  48. let container = try decoder.container(keyedBy: CodingKeys.self)
  49. timestamp = try container.decode(Date.self, forKey: .timestamp)
  50. category = try container.decode(String.self, forKey: .category)
  51. name = try container.decode(String.self, forKey: .name)
  52. tasks = try container.decode([Task].self, forKey: .tasks)
  53. tasks.forEach({ $0.project = self })
  54. }
  55. func encode(to encoder: any Encoder) throws {
  56. var container = encoder.container(keyedBy: CodingKeys.self)
  57. try container.encode(timestamp, forKey: .timestamp)
  58. try container.encode(category, forKey: .category)
  59. try container.encode(tasks, forKey: .tasks)
  60. try container.encode(name, forKey: .name)
  61. }
  62. }