You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

224 lines
8.4 KiB

// Copyright © 2015 Abhishek Banthia
import Cocoa
import CoreLoggerKit
import CoreModelKit
enum ViewType {
case futureSlider
case upcomingEventView
case twelveHour
case sunrise
case showMeetingInMenubar
case showAllDayEventsInMenubar
case showAppInForeground
5 years ago
case appDisplayOptions
case dateInMenubar
case placeInMenubar
case dayInMenubar
case menubarCompactMode
3 years ago
case sync
}
class DataStore: NSObject {
private static var sharedStore = DataStore(with: UserDefaults.standard)
private var userDefaults: UserDefaults!
3 years ago
private var ubiquitousStore: NSUbiquitousKeyValueStore?
6 years ago
// Since these pref can accessed every second, let's cache this
private var shouldDisplayDayInMenubar: Bool = false
private var shouldDisplayDateInMenubar: Bool = false
private static let timeFormatsWithSuffix: Set<NSNumber> = Set([NSNumber(integerLiteral: 0),
NSNumber(integerLiteral: 3),
NSNumber(integerLiteral: 4),
NSNumber(integerLiteral: 6),
NSNumber(integerLiteral: 7)])
class func shared() -> DataStore {
return sharedStore
}
init(with defaults: UserDefaults) {
super.init()
userDefaults = defaults
setupSyncNotification()
}
3 years ago
func setupSyncNotification() {
3 years ago
if shouldDisplay(.sync) {
3 years ago
ubiquitousStore = NSUbiquitousKeyValueStore.default
NotificationCenter.default.addObserver(self,
selector: #selector(ubiquitousKeyValueStoreChanged),
name: NSUbiquitousKeyValueStore.didChangeExternallyNotification,
object: NSUbiquitousKeyValueStore.default)
3 years ago
let synchronizationResult = ubiquitousStore?.synchronize() ?? false
let resultString = synchronizationResult ? "successfully" : "unsuccessfully"
Logger.info("Ubiquitous Store synchronized \(resultString)")
} else {
NotificationCenter.default.removeObserver(self,
name: NSUbiquitousKeyValueStore.didChangeExternallyNotification,
object: nil)
3 years ago
}
}
3 years ago
3 years ago
@objc func ubiquitousKeyValueStoreChanged(_ notification: Notification) {
let userInfo = notification.userInfo ?? [:]
let ubiquitousStore = notification.object as? NSUbiquitousKeyValueStore
Logger.info("Ubiquitous Store Changed: User Info is \(userInfo)")
3 years ago
let currentTimezones = userDefaults.object(forKey: CLDefaultPreferenceKey) as? [Data]
let cloudTimezones = ubiquitousStore?.object(forKey: CLDefaultPreferenceKey) as? [Data]
3 years ago
let cloudLastUpdateDate = (ubiquitousStore?.object(forKey: CLUbiquitousStoreLastUpdateKey) as? Date) ?? Date()
let defaultsLastUpdateDate = (ubiquitousStore?.object(forKey: CLUserDefaultsLastUpdateKey) as? Date) ?? Date()
3 years ago
if cloudTimezones == currentTimezones {
Logger.info("Ubiquitous Store timezones aren't equal to current timezones")
}
3 years ago
if defaultsLastUpdateDate.isLaterThanOrEqual(to: cloudLastUpdateDate) {
Logger.info("Ubiquitous Store is stale as compared to User Defaults")
}
3 years ago
3 years ago
if cloudTimezones != currentTimezones, cloudLastUpdateDate.isLaterThanOrEqual(to: defaultsLastUpdateDate) {
Logger.info("Syncing local timezones with data from the ☁. ☁ last update timestamp is recent")
3 years ago
userDefaults.set(cloudTimezones, forKey: CLDefaultPreferenceKey)
3 years ago
userDefaults.set(Date(), forKey: CLUserDefaultsLastUpdateKey)
3 years ago
NotificationCenter.default.post(name: DataStore.didSyncFromExternalSourceNotification,
object: self)
return
3 years ago
}
}
func timezones() -> [Data] {
guard let preferences = userDefaults.object(forKey: CLDefaultPreferenceKey) as? [Data] else {
return []
}
return preferences
}
3 years ago
func setTimezones(_ timezones: [Data]?) {
userDefaults.set(timezones, forKey: CLDefaultPreferenceKey)
3 years ago
userDefaults.set(Date(), forKey: CLUserDefaultsLastUpdateKey)
// iCloud sync
3 years ago
ubiquitousStore?.set(timezones, forKey: CLDefaultPreferenceKey)
3 years ago
ubiquitousStore?.set(Date(), forKey: CLUbiquitousStoreLastUpdateKey)
}
6 years ago
func menubarTimezones() -> [Data]? {
return timezones().filter {
let customTimezone = TimezoneData.customObject(from: $0)
return customTimezone?.isFavourite == 1
}
}
// MARK: Date (May 8th) in Compact Menubar
func shouldShowDateInMenubar() -> Bool {
return shouldDisplay(.dateInMenubar)
}
// MARK: Day (Sun, Mon etc.) in Compact Menubar
func shouldShowDayInMenubar() -> Bool {
return shouldDisplay(.dayInMenubar)
}
6 years ago
func retrieve(key: String) -> Any? {
return userDefaults.object(forKey: key)
}
func addTimezone(_ timezone: TimezoneData) {
let encodedTimezone = NSKeyedArchiver.archivedData(withRootObject: timezone)
var defaults: [Data] = timezones()
defaults.append(encodedTimezone)
setTimezones(defaults)
}
6 years ago
func removeLastTimezone() {
var currentLineup = timezones()
6 years ago
if currentLineup.isEmpty {
return
}
6 years ago
currentLineup.removeLast()
6 years ago
Logger.log(object: [:], for: "Undo Action Executed during Onboarding")
6 years ago
setTimezones(currentLineup)
}
6 years ago
func timezoneFormat() -> NSNumber {
return userDefaults.object(forKey: CLSelectedTimeZoneFormatKey) as? NSNumber ?? NSNumber(integerLiteral: 0)
}
func isBufferRequiredForTwelveHourFormats() -> Bool {
return DataStore.timeFormatsWithSuffix.contains(timezoneFormat())
}
func shouldDisplay(_ type: ViewType) -> Bool {
switch type {
case .futureSlider:
guard let value = retrieve(key: CLDisplayFutureSliderKey) as? NSNumber else {
return false
}
return value != 2 // Modern is 0, Legacy is 1 and Hide is 2.
case .upcomingEventView:
guard let value = retrieve(key: CLShowUpcomingEventView) as? NSString else {
return false
}
return value == "YES"
case .twelveHour:
return shouldDisplayHelper(CLSelectedTimeZoneFormatKey)
case .showAllDayEventsInMenubar:
6 years ago
return shouldDisplayHelper(CLShowAllDayEventsInUpcomingView)
case .sunrise:
6 years ago
return shouldDisplayHelper(CLSunriseSunsetTime)
case .showMeetingInMenubar:
6 years ago
return shouldDisplayHelper(CLShowMeetingInMenubar)
case .showAppInForeground:
guard let value = retrieve(key: CLShowAppInForeground) as? NSNumber else {
return false
}
return value.isEqual(to: NSNumber(value: 1))
case .dateInMenubar:
return shouldDisplayNonObjectHelper(CLShowDateInMenu)
case .placeInMenubar:
6 years ago
return shouldDisplayHelper(CLShowPlaceInMenu)
case .dayInMenubar:
return shouldDisplayNonObjectHelper(CLShowDayInMenu)
5 years ago
case .appDisplayOptions:
4 years ago
return shouldDisplayHelper(CLAppDisplayOptions)
case .menubarCompactMode:
guard let value = retrieve(key: CLMenubarCompactMode) as? Int else {
return false
}
6 years ago
return value == 0
3 years ago
case .sync:
return shouldDisplayHelper(CLEnableSyncKey)
}
}
3 years ago
// MARK: Private
private func shouldDisplayHelper(_ key: String) -> Bool {
guard let value = retrieve(key: key) as? NSNumber else {
return false
}
return value.isEqual(to: NSNumber(value: 0))
}
// MARK: Some values are stored as plain integers; objectForKey: will return nil, so using integerForKey:
private func shouldDisplayNonObjectHelper(_ key: String) -> Bool {
let value = userDefaults.integer(forKey: key)
return value == 0
}
}
3 years ago
extension DataStore {
public static let didSyncFromExternalSourceNotification: NSNotification.Name = .init("didSyncFromExternalSourceNotification")
}