Add todo CRUD tab in companion app

This commit is contained in:
2026-01-10 20:20:32 +00:00
parent c13a4f3247
commit 5af6d38e0f
5 changed files with 417 additions and 0 deletions

View File

@@ -0,0 +1,31 @@
//
// TodoItem.swift
// iris
//
// Created by Codex.
//
import Foundation
struct TodoItem: Identifiable, Codable, Equatable {
let id: UUID
var title: String
var isCompleted: Bool
var createdAt: Date
var updatedAt: Date
var completedAt: Date?
init(id: UUID = UUID(),
title: String,
isCompleted: Bool = false,
createdAt: Date = Date(),
updatedAt: Date = Date(),
completedAt: Date? = nil) {
self.id = id
self.title = title
self.isCompleted = isCompleted
self.createdAt = createdAt
self.updatedAt = updatedAt
self.completedAt = completedAt
}
}

View File

@@ -0,0 +1,116 @@
//
// TodoStore.swift
// iris
//
// Created by Codex.
//
import Foundation
final class TodoStore {
private struct Persisted: Codable {
var schema: Int?
var items: [TodoItem]
}
private let queue = DispatchQueue(label: "iris.todostore.queue")
private let fileURL: URL
private var items: [TodoItem]
init(filename: String = "todos_v1.json") {
self.fileURL = Self.defaultFileURL(filename: filename)
let persisted = Self.load(from: fileURL)
self.items = persisted?.items ?? []
}
func snapshot() -> [TodoItem] {
queue.sync { items }
}
@discardableResult
func add(title: String, now: Date = Date()) -> TodoItem {
queue.sync {
let item = TodoItem(title: title, createdAt: now, updatedAt: now)
items.append(item)
save()
return item
}
}
@discardableResult
func updateTitle(id: UUID, title: String, now: Date = Date()) -> TodoItem? {
queue.sync {
guard let index = items.firstIndex(where: { $0.id == id }) else { return nil }
var item = items[index]
item.title = title
item.updatedAt = now
items[index] = item
save()
return item
}
}
@discardableResult
func setCompleted(id: UUID, completed: Bool, now: Date = Date()) -> TodoItem? {
queue.sync {
guard let index = items.firstIndex(where: { $0.id == id }) else { return nil }
var item = items[index]
item.isCompleted = completed
item.updatedAt = now
item.completedAt = completed ? now : nil
items[index] = item
save()
return item
}
}
func delete(id: UUID) {
queue.sync {
items.removeAll { $0.id == id }
save()
}
}
private func save() {
let persisted = Persisted(schema: 1, items: items)
Self.save(persisted, to: fileURL)
}
private static func defaultFileURL(filename: String) -> URL {
let fm = FileManager.default
let base = fm.urls(for: .applicationSupportDirectory, in: .userDomainMask).first ?? fm.temporaryDirectory
let bundle = Bundle.main.bundleIdentifier ?? "iris"
return base
.appendingPathComponent(bundle, isDirectory: true)
.appendingPathComponent(filename, isDirectory: false)
}
private static func load(from url: URL) -> Persisted? {
guard let data = try? Data(contentsOf: url) else { return nil }
return try? makeDecoder().decode(Persisted.self, from: data)
}
private static func save(_ persisted: Persisted, to url: URL) {
do {
let fm = FileManager.default
try fm.createDirectory(at: url.deletingLastPathComponent(), withIntermediateDirectories: true)
let data = try makeEncoder().encode(persisted)
try data.write(to: url, options: [.atomic])
} catch {
// Best-effort persistence.
}
}
private static func makeEncoder() -> JSONEncoder {
let encoder = JSONEncoder()
encoder.outputFormatting = [.prettyPrinted, .withoutEscapingSlashes]
encoder.dateEncodingStrategy = .iso8601
return encoder
}
private static func makeDecoder() -> JSONDecoder {
let decoder = JSONDecoder()
decoder.dateDecodingStrategy = .iso8601
return decoder
}
}