Widget Extension Fixes: - Create standalone WidgetDataProvider for widget data isolation - Add WIDGET_EXTENSION compiler flag for conditional compilation - Fix DataController references in widget-shared files - Sync widget version numbers with main app (23, 1.0.2) - Add WidgetBackground color to asset catalog Warning Resolutions: - Fix UIScreen.main deprecation in BGView and SharingListView - Fix Text '+' concatenation deprecation in PurchaseButtonView and SettingsTabView - Fix exhaustive switch in BiometricAuthManager (add .none case) - Fix var to let in ExportService (3 instances) - Fix unused result warning in NoteEditorView - Fix ForEach duplicate ID warnings in MonthView and YearView Code Quality Improvements: - Wrap bypassSubscription in #if DEBUG for security - Rename StupidAssCustomWidgetObservableObject to CustomWidgetStateViewModel - Add @MainActor to IconViewModel - Replace fatalError with graceful fallback in SharedModelContainer - Add [weak self] to closures in DayViewViewModel - Add OSLog-based AppLogger for production logging - Add ImageCache with NSCache for memory efficiency - Add AccessibilityHelpers with Reduce Motion support - Create DataControllerProtocol for dependency injection - Update .gitignore with secrets exclusions 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
85 lines
2.4 KiB
Swift
85 lines
2.4 KiB
Swift
//
|
|
// DataController.swift
|
|
// Feels
|
|
//
|
|
// SwiftData controller replacing Core Data PersistenceController.
|
|
//
|
|
|
|
import SwiftData
|
|
import SwiftUI
|
|
import os.log
|
|
|
|
@MainActor
|
|
final class DataController: ObservableObject {
|
|
private static let logger = Logger(subsystem: Bundle.main.bundleIdentifier ?? "com.tt.ifeel", category: "DataController")
|
|
static let shared = DataController()
|
|
|
|
private(set) var container: ModelContainer
|
|
|
|
var modelContext: ModelContext {
|
|
container.mainContext
|
|
}
|
|
|
|
private var useCloudKit: Bool {
|
|
GroupUserDefaults.groupDefaults.bool(forKey: UserDefaultsStore.Keys.useCloudKit.rawValue)
|
|
}
|
|
|
|
// Listeners for data changes (keeping existing pattern)
|
|
var switchContainerListeners = [(() -> Void)]()
|
|
private var editedDataClosure = [() -> Void]()
|
|
|
|
// Computed properties for earliest/latest entries
|
|
var earliestEntry: MoodEntryModel? {
|
|
var descriptor = FetchDescriptor<MoodEntryModel>(
|
|
sortBy: [SortDescriptor(\.forDate, order: .forward)]
|
|
)
|
|
descriptor.fetchLimit = 1
|
|
return try? modelContext.fetch(descriptor).first
|
|
}
|
|
|
|
var latestEntry: MoodEntryModel? {
|
|
var descriptor = FetchDescriptor<MoodEntryModel>(
|
|
sortBy: [SortDescriptor(\.forDate, order: .reverse)]
|
|
)
|
|
descriptor.fetchLimit = 1
|
|
return try? modelContext.fetch(descriptor).first
|
|
}
|
|
|
|
private init() {
|
|
let cloudKit = GroupUserDefaults.groupDefaults.bool(forKey: UserDefaultsStore.Keys.useCloudKit.rawValue)
|
|
container = SharedModelContainer.createWithFallback(useCloudKit: cloudKit)
|
|
}
|
|
|
|
// MARK: - Container Switching (for CloudKit toggle)
|
|
|
|
func switchContainer() {
|
|
save()
|
|
container = SharedModelContainer.createWithFallback(useCloudKit: useCloudKit)
|
|
for listener in switchContainerListeners {
|
|
listener()
|
|
}
|
|
}
|
|
|
|
// MARK: - Listener Management
|
|
|
|
func addNewDataListener(closure: @escaping (() -> Void)) {
|
|
editedDataClosure.append(closure)
|
|
}
|
|
|
|
func saveAndRunDataListeners() {
|
|
save()
|
|
for closure in editedDataClosure {
|
|
closure()
|
|
}
|
|
}
|
|
|
|
func save() {
|
|
guard modelContext.hasChanges else { return }
|
|
do {
|
|
try modelContext.save()
|
|
} catch {
|
|
Self.logger.error("Failed to save context: \(error.localizedDescription)")
|
|
}
|
|
}
|
|
}
|