Widget anpassung vorbereitung
This commit is contained in:
135
Shared/WidgetData.swift
Normal file
135
Shared/WidgetData.swift
Normal file
@@ -0,0 +1,135 @@
|
||||
import Foundation
|
||||
#if canImport(WidgetKit)
|
||||
import WidgetKit
|
||||
#endif
|
||||
|
||||
/// App-Group identifier shared between the main app and the widget extension.
|
||||
/// IMPORTANT: This must match the App Group capability in BOTH targets
|
||||
/// and the App Group ID registered in the Apple Developer Portal.
|
||||
let widgetAppGroupID = "group.com.scarriffleservices.calendarr"
|
||||
|
||||
/// Lightweight event representation that lives inside the widget cache.
|
||||
/// We strip everything the widget doesn't need (notes, calendar IDs, URLs).
|
||||
struct WidgetEvent: Codable, Hashable, Identifiable {
|
||||
let id: String
|
||||
let title: String
|
||||
let start: Date
|
||||
let end: Date
|
||||
let isAllDay: Bool
|
||||
let colorHex: String
|
||||
let location: String
|
||||
}
|
||||
|
||||
/// Snapshot blob the app writes to the App-Group container and the widget reads.
|
||||
struct WidgetSnapshot: Codable {
|
||||
let writtenAt: Date
|
||||
let events: [WidgetEvent]
|
||||
/// Mirrors the user's chosen visual settings so the widget looks the same
|
||||
/// as the app even when its own AppStorage in the extension is empty.
|
||||
let todayColorHex: String
|
||||
let textColorHex: String
|
||||
let backgroundColorHex: String
|
||||
let lineColorHex: String
|
||||
let primaryColorHex: String
|
||||
let accentColorHex: String
|
||||
let language: String
|
||||
|
||||
init(writtenAt: Date,
|
||||
events: [WidgetEvent],
|
||||
todayColorHex: String,
|
||||
textColorHex: String,
|
||||
backgroundColorHex: String,
|
||||
lineColorHex: String,
|
||||
primaryColorHex: String,
|
||||
accentColorHex: String,
|
||||
language: String) {
|
||||
self.writtenAt = writtenAt
|
||||
self.events = events
|
||||
self.todayColorHex = todayColorHex
|
||||
self.textColorHex = textColorHex
|
||||
self.backgroundColorHex = backgroundColorHex
|
||||
self.lineColorHex = lineColorHex
|
||||
self.primaryColorHex = primaryColorHex
|
||||
self.accentColorHex = accentColorHex
|
||||
self.language = language
|
||||
}
|
||||
|
||||
/// Custom decoder so older caches without the new colour fields still load.
|
||||
init(from decoder: Decoder) throws {
|
||||
let c = try decoder.container(keyedBy: CodingKeys.self)
|
||||
writtenAt = try c.decode(Date.self, forKey: .writtenAt)
|
||||
events = try c.decode([WidgetEvent].self, forKey: .events)
|
||||
todayColorHex = try c.decode(String.self, forKey: .todayColorHex)
|
||||
textColorHex = try c.decode(String.self, forKey: .textColorHex)
|
||||
backgroundColorHex = try c.decode(String.self, forKey: .backgroundColorHex)
|
||||
lineColorHex = try c.decode(String.self, forKey: .lineColorHex)
|
||||
language = try c.decode(String.self, forKey: .language)
|
||||
primaryColorHex = try c.decodeIfPresent(String.self, forKey: .primaryColorHex) ?? "#4285f4"
|
||||
accentColorHex = try c.decodeIfPresent(String.self, forKey: .accentColorHex) ?? "#ea4335"
|
||||
}
|
||||
|
||||
private enum CodingKeys: String, CodingKey {
|
||||
case writtenAt, events, todayColorHex, textColorHex, backgroundColorHex
|
||||
case lineColorHex, primaryColorHex, accentColorHex, language
|
||||
}
|
||||
}
|
||||
|
||||
enum WidgetStore {
|
||||
private static let cacheFilename = "widget-cache.json"
|
||||
|
||||
private static var containerURL: URL? {
|
||||
FileManager.default.containerURL(forSecurityApplicationGroupIdentifier: widgetAppGroupID)
|
||||
}
|
||||
|
||||
private static var cacheURL: URL? {
|
||||
containerURL?.appendingPathComponent(cacheFilename)
|
||||
}
|
||||
|
||||
/// Called by the app whenever the event cache changes.
|
||||
static func write(_ snapshot: WidgetSnapshot) {
|
||||
guard let url = cacheURL else { return }
|
||||
let encoder = JSONEncoder()
|
||||
encoder.dateEncodingStrategy = .iso8601
|
||||
if let data = try? encoder.encode(snapshot) {
|
||||
try? data.write(to: url, options: .atomic)
|
||||
}
|
||||
}
|
||||
|
||||
/// Called by the widget timeline provider to load the latest snapshot.
|
||||
static func read() -> WidgetSnapshot? {
|
||||
guard let url = cacheURL, let data = try? Data(contentsOf: url) else { return nil }
|
||||
let decoder = JSONDecoder()
|
||||
decoder.dateDecodingStrategy = .iso8601
|
||||
return try? decoder.decode(WidgetSnapshot.self, from: data)
|
||||
}
|
||||
|
||||
/// Rewrite the existing snapshot with the latest colour / language values
|
||||
/// from UserDefaults. Used when the user tweaks an appearance setting and
|
||||
/// we want the widgets to refresh immediately, without needing a new event
|
||||
/// sync. No-op if there's no cached snapshot yet.
|
||||
static func republishAppearanceOnly() {
|
||||
guard let existing = read() else { return }
|
||||
let defaults = UserDefaults.standard
|
||||
let updated = WidgetSnapshot(
|
||||
writtenAt: Date(),
|
||||
events: existing.events,
|
||||
todayColorHex: defaults.string(forKey: "todayColor") ?? existing.todayColorHex,
|
||||
textColorHex: defaults.string(forKey: "textColor") ?? existing.textColorHex,
|
||||
backgroundColorHex: defaults.string(forKey: "backgroundColor") ?? existing.backgroundColorHex,
|
||||
lineColorHex: defaults.string(forKey: "lineColor") ?? existing.lineColorHex,
|
||||
primaryColorHex: defaults.string(forKey: "primaryColor") ?? existing.primaryColorHex,
|
||||
accentColorHex: defaults.string(forKey: "accentColor") ?? existing.accentColorHex,
|
||||
language: defaults.string(forKey: "appLanguage") ?? existing.language
|
||||
)
|
||||
write(updated)
|
||||
WidgetTimelineNotifier.reload()
|
||||
}
|
||||
}
|
||||
|
||||
enum WidgetTimelineNotifier {
|
||||
static func reload() {
|
||||
#if canImport(WidgetKit)
|
||||
WidgetCenter.shared.reloadAllTimelines()
|
||||
#endif
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user