// Copyright © 2015 Abhishek Banthia import Cocoa import CoreLocation import CoreLoggerKit import CoreModelKit class TimezoneDataOperations: NSObject { private let dataObject: TimezoneData private let store: DataStore private lazy var nsCalendar = Calendar.autoupdatingCurrent private static var gregorianCalendar = NSCalendar(calendarIdentifier: NSCalendar.Identifier.gregorian) private static var swiftyCalendar = Calendar(identifier: .gregorian) private static let currentLocale = Locale.current.identifier init(with timezone: TimezoneData, store: DataStore) { dataObject = timezone self.store = store super.init() } } extension TimezoneDataOperations { func time(with sliderValue: Int) -> String { guard let newDate = TimezoneDataOperations.gregorianCalendar?.date(byAdding: .minute, value: sliderValue, to: Date(), options: .matchFirst) else { assertionFailure("Data was unexpectedly nil") return CLEmptyString } if dataObject.timezoneFormat(store.timezoneFormat()) == DateFormat.epochTime { let timezone = TimeZone(identifier: dataObject.timezone()) let offset = timezone?.secondsFromGMT(for: newDate) ?? 0 let value = Int(Date().timeIntervalSince1970 + Double(offset)) return "\(value)" } let dateFormatter = DateFormatterManager.dateFormatterWithFormat(with: .none, format: dataObject.timezoneFormat(store.timezoneFormat()), timezoneIdentifier: dataObject.timezone(), locale: Locale.autoupdatingCurrent) return dateFormatter.string(from: newDate) } func nextDaylightSavingsTransitionIfAvailable(with sliderValue: Int) -> String? { let currentTimezone = TimeZone(identifier: dataObject.timezone()) guard let nextDaylightSavingsTransition = currentTimezone?.nextDaylightSavingTimeTransition else { return nil } guard let newDate = TimezoneDataOperations.gregorianCalendar?.date(byAdding: .minute, value: sliderValue, to: Date(), options: .matchFirst) else { assertionFailure("Data was unexpectedly nil") return nil } let calendar = Calendar.autoupdatingCurrent let numberOfDays = nextDaylightSavingsTransition.days(from: newDate, calendar: calendar) // We'd like to show upcoming DST changes within the 7 day range. // Using 8 as a fail-safe as timezones behind CDT can sometimes be wrongly attributed if numberOfDays > 8 || numberOfDays < 0 { return nil } if numberOfDays == 0 { let hoursLeft = nextDaylightSavingsTransition.hours(from: newDate) let suffix = hoursLeft == 1 ? "hour" : "hours" return "Heads up! DST transition will occur in \(hoursLeft) \(suffix)." } let suffix = numberOfDays == 1 ? "day" : "days" return "Heads up! DST transition will occur in \(numberOfDays) \(suffix)." } func compactMenuTitle() -> String { var subtitle = CLEmptyString let shouldDayBeShown = store.shouldShowDayInMenubar() let shouldLabelBeShownAlongWithTime = !store.shouldDisplay(.placeInMenubar) if shouldDayBeShown, shouldLabelBeShownAlongWithTime { let substring = date(with: 0, displayType: .menu) subtitle.append(substring) } let shouldDateBeShown = store.shouldShowDateInMenubar() if shouldDateBeShown, shouldLabelBeShownAlongWithTime { let date = Date().formatter(with: "MMM d", timeZone: dataObject.timezone()) subtitle.isEmpty ? subtitle.append("\(date)") : subtitle.append(" \(date)") } return subtitle.isEmpty ? dataObject.formattedTimezoneLabel() : subtitle } func compactMenuSubtitle() -> String { var subtitle = CLEmptyString let shouldDayBeShown = store.shouldShowDayInMenubar() let shouldLabelsNotBeShownAlongWithTime = store.shouldDisplay(.placeInMenubar) if shouldDayBeShown, shouldLabelsNotBeShownAlongWithTime { let substring = date(with: 0, displayType: .menu) subtitle.append(substring) } let shouldDateBeShown = store.shouldShowDateInMenubar() if shouldDateBeShown, shouldLabelsNotBeShownAlongWithTime { let date = Date().formatter(with: "MMM d", timeZone: dataObject.timezone()) subtitle.isEmpty ? subtitle.append("\(date)") : subtitle.append(" \(date)") } subtitle.isEmpty ? subtitle.append(time(with: 0)) : subtitle.append(" \(time(with: 0))") return subtitle } func menuTitle() -> String { var menuTitle = CLEmptyString let dataStore = store let shouldCityBeShown = dataStore.shouldDisplay(.placeInMenubar) let shouldDayBeShown = dataStore.shouldShowDayInMenubar() let shouldDateBeShown = dataStore.shouldShowDateInMenubar() if shouldCityBeShown { if let address = dataObject.formattedAddress, address.isEmpty == false { if let label = dataObject.customLabel { label.isEmpty == false ? menuTitle.append(label) : menuTitle.append(address) } else { menuTitle.append(address) } } else { if let label = dataObject.customLabel { label.isEmpty == false ? menuTitle.append(label) : menuTitle.append(dataObject.timezone()) } else { menuTitle.append(dataObject.timezone()) } } } if shouldDayBeShown { var substring = date(with: 0, displayType: .menu) if substring.count > 3 { let endIndex = substring.index(substring.startIndex, offsetBy: 2) substring = String(substring[substring.startIndex ... endIndex]) } if menuTitle.isEmpty == false { menuTitle.append(" \(substring.capitalized)") } else { menuTitle.append(substring.capitalized) } } if shouldDateBeShown { let date = Date().formatter(with: "MMM d", timeZone: dataObject.timezone()) if menuTitle.isEmpty == false { menuTitle.append(" \(date)") } else { menuTitle.append("\(date)") } } menuTitle.isEmpty == false ? menuTitle.append(" \(time(with: 0))") : menuTitle.append(time(with: 0)) return menuTitle } private func timezoneDate(with sliderValue: Int, _ calendar: Calendar) -> Date { let source = timezoneDateByAdding(minutesToAdd: sliderValue, calendar) let sourceTimezone = TimeZone.current let destinationTimezone = TimeZone(identifier: dataObject.timezone()) let sourceGMTOffset = Double(sourceTimezone.secondsFromGMT(for: source)) let destinationGMTOffset = Double(destinationTimezone?.secondsFromGMT(for: source) ?? 0) let interval = destinationGMTOffset - sourceGMTOffset return Date(timeInterval: interval, since: source) } // calendar.dateByAdding takes a 0.1% or 0.2% according to TimeProfiler // Let's not use it unless neccesary! private func timezoneDateByAdding(minutesToAdd: Int, _ calendar: Calendar?) -> Date { if minutesToAdd == 0 { return Date() } return calendar?.date(byAdding: .minute, value: minutesToAdd, to: Date()) ?? Date() } func date(with sliderValue: Int, displayType: TimezoneData.DateDisplayType) -> String { guard let relativeDayPreference = store.retrieve(key: CLRelativeDateKey) as? NSNumber else { assertionFailure("Data was unexpectedly nil") return CLEmptyString } if relativeDayPreference.intValue == 3 { return CLEmptyString } var currentCalendar = Calendar(identifier: .gregorian) currentCalendar.locale = Locale.autoupdatingCurrent let convertedDate = timezoneDate(with: sliderValue, currentCalendar) if displayType == .panel { // Yesterday, tomorrow, etc if relativeDayPreference.intValue == 0 { let localFormatter = DateFormatterManager.localizedSimpleFormatter("EEEE") let local = localFormatter.date(from: localeDate(with: "EEEE")) // Gets local week day number and timezone's week day number for comparison let weekDay = currentCalendar.component(.weekday, from: local!) let timezoneWeekday = currentCalendar.component(.weekday, from: convertedDate) if weekDay == timezoneWeekday + 1 { return "Yesterday\(timeDifference())" } else if weekDay == timezoneWeekday { return "Today\(timeDifference())" } else if weekDay + 1 == timezoneWeekday || weekDay - 6 == timezoneWeekday { return "Tomorrow\(timeDifference())" } else { return "\(weekdayText(from: convertedDate))\(timeDifference())" } } // Day name: Thursday, Friday etc if relativeDayPreference.intValue == 1 { return "\(weekdayText(from: convertedDate))\(timeDifference())" } // Date in mmm/dd if relativeDayPreference.intValue == 2 { return "\(todaysDate(with: sliderValue))\(timeDifference())" } let errorDictionary: [String: Any] = ["Timezone": dataObject.timezone(), "Current Locale": Locale.autoupdatingCurrent.identifier, "Slider Value": sliderValue, "Today's Date": Date()] Logger.log(object: errorDictionary, for: "Unable to get date") return "Error" } else { return "\(shortWeekdayText(convertedDate))" } } // Returns shortened weekday given a date // For eg. Thu or Thursday, Tues for Tuesday etc private func shortWeekdayText(_ date: Date) -> String { let localizedFormatter = DateFormatterManager.localizedSimpleFormatter("E") return localizedFormatter.string(from: date) } // Returns proper weekday given a date // For eg. Thursday, Sunday, Friday etc private func weekdayText(from date: Date) -> String { let dateFormatter = DateFormatterManager.localizedFormatter(with: "EEEE", for: TimeZone.current.identifier) return dateFormatter.string(from: date) } // Exposed to public for tests! public func timeDifference() -> String { let localFormatter = DateFormatterManager.localizedSimpleFormatter("d MMM yyyy HH:mm:ss") let local = localFormatter.date(from: localeDate(with: "d MMM yyyy HH:mm:ss"))! let newDate = timezoneDateByAdding(minutesToAdd: 0, TimezoneDataOperations.swiftyCalendar) let dateFormatter = DateFormatterManager.localizedFormatter(with: "d MMM yyyy HH:mm:ss", for: dataObject.timezone()) guard let timezoneDate = localFormatter.date(from: dateFormatter.string(from: newDate)) else { let unableToConvertDateParameters = [ "New Date": newDate, "Timezone": dataObject.timezone(), "Locale": dateFormatter.locale.identifier, ] as [String: Any] Logger.log(object: unableToConvertDateParameters, for: "Date conversion failure - New Date is nil") return CLEmptyString } let timeDifference = local.timeAgo(since: timezoneDate) if timeDifference.isEmpty { return CLEmptyString } if (local as NSDate).earlierDate(timezoneDate) == local { var replaceAgo = CLEmptyString replaceAgo.append(", +") let agoString = timezoneDate.timeAgo(since: local, numericDates: true) replaceAgo.append(agoString.replacingOccurrences(of: "ago", with: CLEmptyString)) if !TimezoneDataOperations.currentLocale.contains("en") { if TimezoneDataOperations.currentLocale.contains("de") { replaceAgo = replaceAgo.replacingOccurrences(of: "Vor ", with: CLEmptyString) replaceAgo.append(" vor") } return replaceAgo } let minuteDifference = calculateTimeDifference(with: local as NSDate, timezoneDate: timezoneDate as NSDate) minuteDifference == 0 ? replaceAgo.append("") : replaceAgo.append("\(minuteDifference)m") return replaceAgo.lowercased() } var replaceAgo = CLEmptyString replaceAgo.append(", -") let replaced = timeDifference.replacingOccurrences(of: "ago", with: CLEmptyString) replaceAgo.append(replaced) if !TimezoneDataOperations.currentLocale.contains("en") { if TimezoneDataOperations.currentLocale.contains("de") { replaceAgo = replaceAgo.replacingOccurrences(of: "Vor ", with: CLEmptyString) replaceAgo.append(" zurück") } return replaceAgo } let minuteDifference = calculateTimeDifference(with: local as NSDate, timezoneDate: timezoneDate as NSDate) minuteDifference == 0 ? replaceAgo.append("") : replaceAgo.append("\(minuteDifference)m") return replaceAgo.lowercased() } private func initializeSunriseSunset(with sliderValue: Int) { let currentDate = nsCalendar.date(byAdding: .minute, value: sliderValue, to: Date()) guard let lat = dataObject.latitude, let long = dataObject.longitude else { assertionFailure("Data was unexpectedly nil.") return } let coordinates = CLLocationCoordinate2D(latitude: lat, longitude: long) guard let dateForCalculation = currentDate, let solar = Solar(for: dateForCalculation, coordinate: coordinates) else { return } if let sunrise = solar.sunrise, let sunset = solar.sunset { dataObject.sunriseTime = sunrise dataObject.sunsetTime = sunset dataObject.isSunriseOrSunset = solar.isNighttime } else { Logger.log(object: ["Unable to fetch sunrise/sunset": dataObject.formattedTimezoneLabel()], for: "Sunrise/Sunset Error") } } private func calculateTimeDifference(with localDate: NSDate, timezoneDate: NSDate) -> Int { let earliest = localDate.earlierDate(timezoneDate as Date) let latest = earliest == localDate as Date ? timezoneDate : localDate // if timeAgo < 24h => compare DateTime else compare Date only let upToHours: Set = [.second, .minute, .hour] let difference = nsCalendar.dateComponents(upToHours, from: earliest, to: latest as Date) return difference.minute! } func formattedSunriseTime(with sliderValue: Int) -> String { /* We have to call this everytime so that we get an updated value everytime! */ if dataObject.selectionType == .timezone || (dataObject.latitude == nil || dataObject.longitude == nil) { return CLEmptyString } initializeSunriseSunset(with: sliderValue) if let sunrise = dataObject.sunriseTime, let sunset = dataObject.sunsetTime { let correct = dataObject.isSunriseOrSunset ? sunrise : sunset let dateFormatter = DateFormatter() dateFormatter.locale = Locale(identifier: "en_US") dateFormatter.timeZone = TimeZone(identifier: dataObject.timezone()) dateFormatter.dateFormat = dataObject.timezoneFormat(store.timezoneFormat()) return dateFormatter.string(from: correct) } return CLEmptyString } func todaysDate(with sliderValue: Int, locale: Locale = Locale(identifier: "en-US")) -> String { let newDate = TimezoneDataOperations.gregorianCalendar?.date(byAdding: .minute, value: sliderValue, to: Date(), options: .matchFirst) let date = newDate!.formatter(with: "MMM d", timeZone: dataObject.timezone(), locale: locale) return date } private func localDate() -> String { let dateFormatter = DateFormatterManager.dateFormatter(with: .medium, for: TimeZone.autoupdatingCurrent.identifier) return dateFormatter.string(from: Date()) } private func localeDate(with format: String) -> String { let dateFormatter = DateFormatterManager.localizedFormatter(with: format, for: TimeZone.autoupdatingCurrent.identifier) return dateFormatter.string(from: Date()) } func saveObject(at index: Int = -1) { var defaults = store.timezones() guard let encodedObject = NSKeyedArchiver.clocker_archive(with:dataObject as Any) else { return } index == -1 ? defaults.append(encodedObject) : defaults.insert(encodedObject, at: index) store.setTimezones(defaults) } } extension Date { func formatter(with format: String, timeZone: String, locale: Locale = Locale(identifier: "en-US")) -> String { let dateFormatter = DateFormatterManager.dateFormatterWithFormat(with: .medium, format: format, timezoneIdentifier: timeZone, locale: locale) return dateFormatter.string(from: self) } }