mirror of
https://github.com/writeas/writefreely-swiftui-multiplatform.git
synced 2024-11-15 01:11:02 +00:00
549ad3039f
* Set "recommended project settings" for Mac app
* Add default fallback for font typingAttribute
* Clean up linter errors
* Update extension iconset
* Fix spm errors and upgrade sparkle (#203)
* Switch WriteFreely package import from ssh to https
* Sign macOS project target for Development
This fixes a crash-on-launch bug when running Product > Run from Xcode (Product > Archive should work correctly). See https://github.com/sparkle-project/Sparkle/issues/2056 for more details.
Bug introduced by commit:
671b0540ac
* Update Sparkle to v2.0 from official repo
Previously, we forked the repo to the WriteFreely org because Sparkle didn't support tag-based SPM updates. We therefore had to pull from the `master` branch, so forking insulated us from unexpected changes.
As of 2.x Sparkle now allows for tag-based updating via SPM; we'll allow patch updates but stop short of 2.1 until we can vet changes.
* Add MacUpdatesViewModel ObservableObject
* Use MacUpdatesViewModel to handle Sparkle update logic
124 lines
4.4 KiB
Swift
124 lines
4.4 KiB
Swift
import CoreData
|
||
|
||
#if os(iOS)
|
||
import UIKit
|
||
#elseif os(macOS)
|
||
import AppKit
|
||
#endif
|
||
|
||
final class LocalStorageManager {
|
||
|
||
public static var standard = LocalStorageManager()
|
||
public let container: NSPersistentContainer
|
||
private let containerName = "LocalStorageModel"
|
||
|
||
private init() {
|
||
container = NSPersistentContainer(name: containerName)
|
||
setupStore(in: container)
|
||
registerObservers()
|
||
}
|
||
|
||
func saveContext() {
|
||
if container.viewContext.hasChanges {
|
||
do {
|
||
try container.viewContext.save()
|
||
} catch {
|
||
print("Error saving context: \(error)")
|
||
}
|
||
}
|
||
}
|
||
|
||
func purgeUserCollections() {
|
||
let fetchRequest: NSFetchRequest<NSFetchRequestResult> = NSFetchRequest(entityName: "WFACollection")
|
||
let deleteRequest = NSBatchDeleteRequest(fetchRequest: fetchRequest)
|
||
|
||
do {
|
||
try container.viewContext.executeAndMergeChanges(using: deleteRequest)
|
||
} catch {
|
||
print("Error: Failed to purge cached collections.")
|
||
}
|
||
}
|
||
|
||
}
|
||
|
||
private extension LocalStorageManager {
|
||
|
||
var oldStoreURL: URL {
|
||
let appSupport = FileManager.default.urls(for: .applicationSupportDirectory, in: .userDomainMask).first!
|
||
return appSupport.appendingPathComponent("LocalStorageModel.sqlite")
|
||
}
|
||
|
||
var sharedStoreURL: URL {
|
||
let id = "group.com.abunchtell.writefreely"
|
||
let groupContainer = FileManager.default.containerURL(forSecurityApplicationGroupIdentifier: id)!
|
||
return groupContainer.appendingPathComponent("LocalStorageModel.sqlite")
|
||
}
|
||
|
||
func setupStore(in container: NSPersistentContainer) {
|
||
if !FileManager.default.fileExists(atPath: oldStoreURL.path) {
|
||
container.persistentStoreDescriptions.first!.url = sharedStoreURL
|
||
}
|
||
|
||
container.loadPersistentStores { _, error in
|
||
if let error = error {
|
||
fatalError("Core Data store failed to load with error: \(error)")
|
||
}
|
||
}
|
||
migrateStore(for: container)
|
||
container.viewContext.automaticallyMergesChangesFromParent = true
|
||
container.viewContext.mergePolicy = NSMergeByPropertyObjectTrumpMergePolicy
|
||
}
|
||
|
||
func migrateStore(for container: NSPersistentContainer) {
|
||
// Check if the shared store exists before attempting a migration — for example, in case we've already attempted
|
||
// and successfully completed a migration, but the deletion of the old store failed for some reason.
|
||
guard !FileManager.default.fileExists(atPath: sharedStoreURL.path) else { return }
|
||
|
||
let coordinator = container.persistentStoreCoordinator
|
||
|
||
// Get a reference to the old store.
|
||
guard let oldStore = coordinator.persistentStore(for: oldStoreURL) else {
|
||
return
|
||
}
|
||
|
||
// Attempt to migrate the old store over to the shared store URL.
|
||
do {
|
||
try coordinator.migratePersistentStore(oldStore,
|
||
to: sharedStoreURL,
|
||
options: nil,
|
||
withType: NSSQLiteStoreType)
|
||
} catch {
|
||
fatalError("Something went wrong migrating the store: \(error)")
|
||
}
|
||
|
||
// Attempt to delete the old store.
|
||
do {
|
||
try FileManager.default.removeItem(at: oldStoreURL)
|
||
} catch {
|
||
fatalError("Something went wrong while deleting the old store: \(error)")
|
||
}
|
||
}
|
||
|
||
func registerObservers() {
|
||
let center = NotificationCenter.default
|
||
|
||
#if os(iOS)
|
||
let notification = UIApplication.willResignActiveNotification
|
||
#elseif os(macOS)
|
||
let notification = NSApplication.willResignActiveNotification
|
||
#endif
|
||
|
||
// We don't need to worry about removing this observer because we're targeting iOS 9+ / macOS 10.11+; the
|
||
// system will clean this up the next time it would be posted to.
|
||
// See: https://developer.apple.com/documentation/foundation/notificationcenter/1413994-removeobserver
|
||
// And: https://developer.apple.com/documentation/foundation/notificationcenter/1407263-removeobserver
|
||
// swiftlint:disable:next discarded_notification_center_observer
|
||
center.addObserver(forName: notification, object: nil, queue: nil, using: self.saveContextOnResignActive)
|
||
}
|
||
|
||
func saveContextOnResignActive(_ notification: Notification) {
|
||
saveContext()
|
||
}
|
||
|
||
}
|