|
|
|
// Copyright © 2015 Abhishek Banthia
|
|
|
|
|
|
|
|
import Cocoa
|
|
|
|
import ServiceManagement
|
|
|
|
|
|
|
|
struct PreferencesConstants {
|
|
|
|
static let timezoneNameIdentifier = "formattedAddress"
|
|
|
|
static let customLabelIdentifier = "label"
|
|
|
|
static let availableTimezoneIdentifier = "availableTimezones"
|
|
|
|
static let noTimezoneSelectedErrorMessage = "Please select a timezone!"
|
|
|
|
static let maxTimezonesErrorMessage = "Maximum 100 timezones allowed!"
|
|
|
|
static let maxCharactersAllowed = "Only 50 characters allowed!"
|
|
|
|
static let noInternetConnectivityError = "You're offline, maybe?"
|
|
|
|
static let tryAgainMessage = "Try again, maybe?"
|
|
|
|
static let offlineErrorMessage = "The Internet connection appears to be offline."
|
|
|
|
}
|
|
|
|
|
|
|
|
class PreferencesViewController: ParentViewController {
|
|
|
|
private var isActivityInProgress = false {
|
|
|
|
didSet {
|
|
|
|
OperationQueue.main.addOperation {
|
|
|
|
self.isActivityInProgress ? self.progressIndicator.startAnimation(nil) : self.progressIndicator.stopAnimation(nil)
|
|
|
|
self.availableTimezoneTableView.isEnabled = !self.isActivityInProgress
|
|
|
|
self.addButton.isEnabled = !self.isActivityInProgress
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private var selectedTimeZones: [Data] {
|
|
|
|
return DataStore.shared().timezones()
|
|
|
|
}
|
|
|
|
|
|
|
|
private var filteredArray: [Any] = []
|
|
|
|
private var timezoneArray: [String] = []
|
|
|
|
private var timezoneFilteredArray: [String] = []
|
|
|
|
private var columnName = "Place(s)"
|
|
|
|
private var dataTask: URLSessionDataTask? = .none
|
|
|
|
|
|
|
|
private lazy var notimezoneView: NoTimezoneView? = {
|
|
|
|
NoTimezoneView(frame: tableview.frame)
|
|
|
|
}()
|
|
|
|
|
|
|
|
// Sorting
|
|
|
|
private var arePlacesSortedInAscendingOrder = false
|
|
|
|
private var arePlacesSortedInAscendingTimezoneOrder = false
|
|
|
|
private var isTimezoneSortOptionSelected = false
|
|
|
|
private var isTimezoneNameSortOptionSelected = false
|
|
|
|
private var isLabelOptionSelected = false
|
|
|
|
|
|
|
|
@IBOutlet private var placeholderLabel: NSTextField!
|
|
|
|
@IBOutlet private var timezoneTableView: NSTableView!
|
|
|
|
@IBOutlet private var availableTimezoneTableView: NSTableView!
|
|
|
|
@IBOutlet private var timezonePanel: Panelr!
|
|
|
|
@IBOutlet private var stackView: NSStackView!
|
|
|
|
@IBOutlet private var progressIndicator: NSProgressIndicator!
|
|
|
|
@IBOutlet private var addButton: NSButton!
|
|
|
|
@IBOutlet private var recorderControl: SRRecorderControl!
|
|
|
|
|
|
|
|
@IBOutlet private var timezoneSortButton: NSButton!
|
|
|
|
@IBOutlet private var timezoneNameSortButton: NSButton!
|
|
|
|
@IBOutlet private var labelSortButton: NSButton!
|
|
|
|
@IBOutlet private var deleteButton: NSButton!
|
|
|
|
@IBOutlet private var addTimezoneButton: NSButton!
|
|
|
|
|
|
|
|
@IBOutlet private var searchField: NSSearchField!
|
|
|
|
@IBOutlet private var messageLabel: NSTextField!
|
|
|
|
@IBOutlet private var searchCriteria: NSSegmentedControl!
|
|
|
|
@IBOutlet private var abbreviation: NSTableColumn!
|
|
|
|
|
|
|
|
@IBOutlet private var headerView: NSView!
|
|
|
|
@IBOutlet private var tableview: NSView!
|
|
|
|
@IBOutlet private var additionalSortOptions: NSView!
|
|
|
|
@IBOutlet var startAtLoginLabel: NSTextField!
|
|
|
|
|
|
|
|
@IBOutlet var startupCheckbox: NSButton!
|
|
|
|
@IBOutlet var headerLabel: NSTextField!
|
|
|
|
|
|
|
|
@IBOutlet var sortToggle: NSButton!
|
|
|
|
private var themeDidChangeNotification: NSObjectProtocol?
|
|
|
|
|
|
|
|
override func viewDidLoad() {
|
|
|
|
super.viewDidLoad()
|
|
|
|
|
|
|
|
NotificationCenter.default.addObserver(self,
|
|
|
|
selector: #selector(refreshTimezoneTableView),
|
|
|
|
name: NSNotification.Name.customLabelChanged,
|
|
|
|
object: nil)
|
|
|
|
|
|
|
|
refreshTimezoneTableView()
|
|
|
|
|
|
|
|
setup()
|
|
|
|
|
|
|
|
availableTimezoneTableView.reloadData()
|
|
|
|
|
|
|
|
setupShortcutObserver()
|
|
|
|
|
|
|
|
darkModeChanges()
|
|
|
|
|
|
|
|
themeDidChangeNotification = NotificationCenter.default.addObserver(forName: .themeDidChangeNotification, object: nil, queue: OperationQueue.main) { _ in
|
|
|
|
self.setup()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
deinit {
|
|
|
|
if let themeDidChangeNotif = themeDidChangeNotification {
|
|
|
|
NotificationCenter.default.removeObserver(themeDidChangeNotif)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func darkModeChanges() {
|
|
|
|
if #available(macOS 10.14, *) {
|
|
|
|
addTimezoneButton.image = NSImage(named: .addDynamicIcon)
|
|
|
|
sortToggle.image = NSImage(named: .sortToggleIcon)
|
|
|
|
sortToggle.alternateImage = NSImage(named: .sortToggleAlternateIcon)
|
|
|
|
deleteButton.image = NSImage(named: NSImage.Name("Remove Dynamic"))!
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func setupLocalizedText() {
|
|
|
|
startAtLoginLabel.stringValue = "Start Clocker at Login"
|
|
|
|
headerLabel.stringValue = "Selected Timezones"
|
|
|
|
timezoneSortButton.title = "Sort by Time Difference"
|
|
|
|
timezoneNameSortButton.title = "Sort by Name"
|
|
|
|
labelSortButton.title = "Sort by Label"
|
|
|
|
}
|
|
|
|
|
|
|
|
@objc func refreshTimezoneTableView() {
|
|
|
|
OperationQueue.main.addOperation {
|
|
|
|
self.build()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func refreshMainTable() {
|
|
|
|
OperationQueue.main.addOperation {
|
|
|
|
self.refresh()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func refresh() {
|
|
|
|
if DataStore.shared().shouldDisplay(ViewType.showAppInForeground) {
|
|
|
|
updateFloatingWindow()
|
|
|
|
} else {
|
|
|
|
guard let panel = PanelController.panel() else { return }
|
|
|
|
panel.updateDefaultPreferences()
|
|
|
|
panel.updateTableContent()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func updateFloatingWindow() {
|
|
|
|
let current = FloatingWindowController.shared()
|
|
|
|
current.updateDefaultPreferences()
|
|
|
|
current.updateTableContent()
|
|
|
|
}
|
|
|
|
|
|
|
|
private func build() {
|
|
|
|
if DataStore.shared().timezones() == [] {
|
|
|
|
housekeeping()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if selectedTimeZones.isEmpty == false {
|
|
|
|
headerView.isHidden = false
|
|
|
|
|
|
|
|
if tableview.subviews.count > 1, let zeroView = notimezoneView, tableview.subviews.contains(zeroView) {
|
|
|
|
zeroView.removeFromSuperview()
|
|
|
|
timezoneTableView.enclosingScrollView?.isHidden = false
|
|
|
|
}
|
|
|
|
timezoneTableView.reloadData()
|
|
|
|
} else {
|
|
|
|
housekeeping()
|
|
|
|
}
|
|
|
|
|
|
|
|
cleanup()
|
|
|
|
}
|
|
|
|
|
|
|
|
private func housekeeping() {
|
|
|
|
timezoneTableView.enclosingScrollView?.isHidden = true
|
|
|
|
headerView.isHidden = true
|
|
|
|
showNoTimezoneState()
|
|
|
|
cleanup()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
private func cleanup() {
|
|
|
|
timezoneTableView.scrollRowToVisible(selectedTimeZones.count - 1)
|
|
|
|
updateMenubarTitles() // Update the menubar titles, the custom labels might have changed.
|
|
|
|
}
|
|
|
|
|
|
|
|
private func updateMenubarTitles() {
|
|
|
|
let defaultTimezones = DataStore.shared().timezones()
|
|
|
|
UserDefaults.standard.set([], forKey: CLMenubarFavorites)
|
|
|
|
|
|
|
|
let menubarTimes = defaultTimezones.compactMap { (data) -> TimezoneData? in
|
|
|
|
if let model = TimezoneData.customObject(from: data), model.isFavourite == 1 {
|
|
|
|
return model
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
let archivedObjects = menubarTimes.map { (timezone) -> Data in
|
|
|
|
NSKeyedArchiver.archivedData(withRootObject: timezone)
|
|
|
|
}
|
|
|
|
|
|
|
|
UserDefaults.standard.set(archivedObjects, forKey: CLMenubarFavorites)
|
|
|
|
|
|
|
|
// Update appereance if in compact menubar mode
|
|
|
|
if let appDelegate = NSApplication.shared.delegate as? AppDelegate {
|
|
|
|
appDelegate.setupMenubarTimer()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func setup() {
|
|
|
|
setupAccessibilityIdentifiers()
|
|
|
|
|
|
|
|
deleteButton.isEnabled = false
|
|
|
|
|
|
|
|
[placeholderLabel, additionalSortOptions].forEach { $0.isHidden = true }
|
|
|
|
|
|
|
|
if timezoneArray.isEmpty {
|
|
|
|
timezoneArray.append("UTC")
|
|
|
|
timezoneArray.append("Anywhere on Earth")
|
|
|
|
timezoneArray.append(contentsOf: NSTimeZone.knownTimeZoneNames)
|
|
|
|
}
|
|
|
|
|
|
|
|
messageLabel.stringValue = CLEmptyString
|
|
|
|
|
|
|
|
timezoneTableView.registerForDraggedTypes([.dragSession])
|
|
|
|
|
|
|
|
progressIndicator.usesThreadedAnimation = true
|
|
|
|
|
|
|
|
setupLocalizedText()
|
|
|
|
|
|
|
|
setupColor()
|
|
|
|
|
|
|
|
startupCheckbox.integerValue = DataStore.shared().retrieve(key: CLStartAtLogin) as? Int ?? 0
|
|
|
|
}
|
|
|
|
|
|
|
|
private func setupColor() {
|
|
|
|
let themer = Themer.shared()
|
|
|
|
|
|
|
|
headerLabel.textColor = themer.mainTextColor()
|
|
|
|
startAtLoginLabel.textColor = Themer.shared().mainTextColor()
|
|
|
|
|
|
|
|
[timezoneNameSortButton, labelSortButton, timezoneSortButton].forEach {
|
|
|
|
$0?.attributedTitle = NSAttributedString(string: $0?.title ?? CLEmptyString, attributes: [
|
|
|
|
NSAttributedString.Key.foregroundColor: Themer.shared().mainTextColor(),
|
|
|
|
NSAttributedString.Key.font: NSFont(name: "Avenir-Light", size: 13)!,
|
|
|
|
])
|
|
|
|
}
|
|
|
|
|
|
|
|
addTimezoneButton.image = themer.addImage()
|
|
|
|
deleteButton.image = themer.removeImage()
|
|
|
|
sortToggle.image = themer.additionalPreferencesImage()
|
|
|
|
sortToggle.alternateImage = themer.additionalPreferencesHighlightedImage()
|
|
|
|
}
|
|
|
|
|
|
|
|
private func setupShortcutObserver() {
|
|
|
|
let defaults = NSUserDefaultsController.shared
|
|
|
|
|
|
|
|
recorderControl.bind(NSBindingName.value,
|
|
|
|
to: defaults,
|
|
|
|
withKeyPath: "values.globalPing",
|
|
|
|
options: nil)
|
|
|
|
|
|
|
|
recorderControl.delegate = self
|
|
|
|
}
|
|
|
|
|
|
|
|
override func observeValue(forKeyPath keyPath: String?, of object: Any?, change _: [NSKeyValueChangeKey: Any]?, context _: UnsafeMutableRawPointer?) {
|
|
|
|
if let path = keyPath, path == "values.globalPing" {
|
|
|
|
let hotKeyCenter = PTHotKeyCenter.shared()
|
|
|
|
let oldHotKey = hotKeyCenter?.hotKey(withIdentifier: path)
|
|
|
|
hotKeyCenter?.unregisterHotKey(oldHotKey)
|
|
|
|
|
|
|
|
guard let newObject = object as? NSObject, let newShortcut = newObject.value(forKeyPath: path) as? [AnyHashable: Any] else {
|
|
|
|
assertionFailure("Unable to recognize shortcuts")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
let newHotKey: PTHotKey = PTHotKey(identifier: keyPath,
|
|
|
|
keyCombo: newShortcut,
|
|
|
|
target: self,
|
|
|
|
action: #selector(ping(_:)))
|
|
|
|
|
|
|
|
hotKeyCenter?.register(newHotKey)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@objc func ping(_ sender: Any) {
|
|
|
|
guard let delegate = NSApplication.shared.delegate as? AppDelegate else {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
delegate.togglePanel(sender)
|
|
|
|
}
|
|
|
|
|
|
|
|
private func showNoTimezoneState() {
|
|
|
|
if let zeroView = notimezoneView {
|
|
|
|
notimezoneView?.wantsLayer = true
|
|
|
|
tableview.addSubview(zeroView)
|
|
|
|
Logger.log(object: ["Showing Empty View": "YES"], for: "Showing Empty View")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func setupAccessibilityIdentifiers() {
|
|
|
|
timezoneTableView.setAccessibilityIdentifier("TimezoneTableView")
|
|
|
|
availableTimezoneTableView.setAccessibilityIdentifier("AvailableTimezoneTableView")
|
|
|
|
searchField.setAccessibilityIdentifier("AvailableSearchField")
|
|
|
|
timezoneSortButton.setAccessibility("SortByDifference")
|
|
|
|
labelSortButton.setAccessibility("SortByLabelButton")
|
|
|
|
timezoneNameSortButton.setAccessibility("SortByTimezoneName")
|
|
|
|
}
|
|
|
|
|
|
|
|
override var acceptsFirstResponder: Bool {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extension PreferencesViewController: NSTableViewDataSource, NSTableViewDelegate {
|
|
|
|
func numberOfRows(in tableView: NSTableView) -> Int {
|
|
|
|
var numberOfRows = 0
|
|
|
|
|
|
|
|
if tableView == timezoneTableView {
|
|
|
|
numberOfRows = selectedTimeZones.count
|
|
|
|
} else {
|
|
|
|
numberOfRows = numberOfSearchResults()
|
|
|
|
}
|
|
|
|
|
|
|
|
return numberOfRows
|
|
|
|
}
|
|
|
|
|
|
|
|
func tableView(_: NSTableView, objectValueFor tableColumn: NSTableColumn?, row: Int) -> Any? {
|
|
|
|
var dataSource: TimezoneData?
|
|
|
|
var selectedDataSource: TimezoneData?
|
|
|
|
|
|
|
|
if filteredArray.count > row, let currentFilteredObject = filteredArray[row] as? TimezoneData {
|
|
|
|
dataSource = currentFilteredObject
|
|
|
|
}
|
|
|
|
|
|
|
|
if selectedTimeZones.count > row, let model = TimezoneData.customObject(from: selectedTimeZones[row]) {
|
|
|
|
selectedDataSource = model
|
|
|
|
}
|
|
|
|
|
|
|
|
if tableColumn?.identifier.rawValue == PreferencesConstants.timezoneNameIdentifier {
|
|
|
|
return handleTimezoneNameIdentifier(for: row, selectedDataSource)
|
|
|
|
}
|
|
|
|
|
|
|
|
if tableColumn?.identifier.rawValue == PreferencesConstants.availableTimezoneIdentifier {
|
|
|
|
return handleAvailableTimezoneColumn(for: row, dataSource)
|
|
|
|
}
|
|
|
|
|
|
|
|
if tableColumn?.identifier.rawValue == PreferencesConstants.customLabelIdentifier {
|
|
|
|
return selectedDataSource?.customLabel ?? "Error"
|
|
|
|
}
|
|
|
|
|
|
|
|
if tableColumn?.identifier.rawValue == "favouriteTimezone" {
|
|
|
|
return selectedDataSource?.isFavourite ?? 0
|
|
|
|
}
|
|
|
|
|
|
|
|
if tableColumn?.identifier.rawValue == "abbreviation" {
|
|
|
|
return handleAbbreviationColumn(for: row)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
private func handleTimezoneNameIdentifier(for _: Int, _ selectedDataSource: TimezoneData?) -> Any? {
|
|
|
|
guard let model = selectedDataSource else {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if let address = model.formattedAddress, address.isEmpty == false {
|
|
|
|
return model.formattedAddress
|
|
|
|
}
|
|
|
|
|
|
|
|
return model.timezoneID
|
|
|
|
}
|
|
|
|
|
|
|
|
private func handleAvailableTimezoneColumn(for row: Int, _ dataSource: TimezoneData?) -> Any? {
|
|
|
|
let criteria = searchCriteria.selectedSegment
|
|
|
|
|
|
|
|
if criteria == 0 {
|
|
|
|
if row < filteredArray.count {
|
|
|
|
return dataSource?.formattedAddress
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if searchField.stringValue.isEmpty == false, row < timezoneFilteredArray.count {
|
|
|
|
return timezoneFilteredArray[row]
|
|
|
|
}
|
|
|
|
return timezoneArray[row]
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
private func handleAbbreviationColumn(for row: Int) -> Any? {
|
|
|
|
if searchField.stringValue.isEmpty == false, row < timezoneFilteredArray.count {
|
|
|
|
let currentSelection = timezoneFilteredArray[row]
|
|
|
|
if currentSelection == "UTC" {
|
|
|
|
return "UTC"
|
|
|
|
} else if currentSelection == "Anywhere on Earth" {
|
|
|
|
return "GMT+12"
|
|
|
|
}
|
|
|
|
|
|
|
|
return NSTimeZone(name: timezoneFilteredArray[row])?.abbreviation ?? "Error"
|
|
|
|
}
|
|
|
|
|
|
|
|
if timezoneArray.count > row {
|
|
|
|
// Special return for manually inserted 'UTC'
|
|
|
|
if timezoneArray[row] == "UTC" {
|
|
|
|
return "UTC"
|
|
|
|
}
|
|
|
|
|
|
|
|
if timezoneArray[row] == "Anywhere on Earth" {
|
|
|
|
return "AoE"
|
|
|
|
}
|
|
|
|
|
|
|
|
return NSTimeZone(name: timezoneArray[row])?.abbreviation ?? "Error"
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func tableView(_: NSTableView, setObjectValue object: Any?, for _: NSTableColumn?, row: Int) {
|
|
|
|
guard !selectedTimeZones.isEmpty, let dataObject = TimezoneData.customObject(from: selectedTimeZones[row]) else {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if let edit = object as? String {
|
|
|
|
setNewLabel(edit, for: dataObject, at: row)
|
|
|
|
} else if let isFavouriteValue = object as? NSNumber {
|
|
|
|
dataObject.isFavourite = isFavouriteValue.intValue
|
|
|
|
insert(timezone: dataObject, at: row)
|
|
|
|
dataObject.isFavourite == 1 ?
|
|
|
|
markAsFavorite(dataObject) :
|
|
|
|
unfavourite(dataObject)
|
|
|
|
updateStatusItem()
|
|
|
|
refreshTimezoneTableView()
|
|
|
|
}
|
|
|
|
|
|
|
|
refreshMainTable()
|
|
|
|
}
|
|
|
|
|
|
|
|
private func markAsFavorite(_ dataObject: TimezoneData) {
|
|
|
|
guard let menubarTitles = DataStore.shared().retrieve(key: CLMenubarFavorites) as? [Data] else {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
var mutableArray = menubarTitles
|
|
|
|
let archivedObject = NSKeyedArchiver.archivedData(withRootObject: dataObject)
|
|
|
|
mutableArray.append(archivedObject)
|
|
|
|
|
|
|
|
UserDefaults.standard.set(mutableArray, forKey: CLMenubarFavorites)
|
|
|
|
|
|
|
|
if dataObject.customLabel != nil {
|
|
|
|
Logger.log(object: ["label": dataObject.customLabel ?? "Error"], for: "favouriteSelected")
|
|
|
|
}
|
|
|
|
|
|
|
|
if let appDelegate = NSApplication.shared.delegate as? AppDelegate {
|
|
|
|
appDelegate.setupMenubarTimer()
|
|
|
|
}
|
|
|
|
|
|
|
|
if mutableArray.count > 1 {
|
|
|
|
showAlertIfMoreThanOneTimezoneHasBeenAddedToTheMenubar()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func unfavourite(_ dataObject: TimezoneData) {
|
|
|
|
guard let menubarTimers = DataStore.shared().retrieve(key: CLMenubarFavorites) as? [Data] else {
|
|
|
|
assertionFailure("Menubar timers is unexpectedly nil")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
Logger.log(object: ["label": dataObject.customLabel ?? "Error"],
|
|
|
|
for: "favouriteRemoved")
|
|
|
|
|
|
|
|
let filteredMenubars = menubarTimers.filter {
|
|
|
|
guard let current = NSKeyedUnarchiver.unarchiveObject(with: $0) as? TimezoneData else {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return current.isEqual(dataObject) == false
|
|
|
|
}
|
|
|
|
|
|
|
|
UserDefaults.standard.set(filteredMenubars, forKey: CLMenubarFavorites)
|
|
|
|
|
|
|
|
if let appDelegate = NSApplication.shared.delegate as? AppDelegate,
|
|
|
|
let menubarFavourites = DataStore.shared().retrieve(key: CLMenubarFavorites) as? [Data],
|
|
|
|
menubarFavourites.isEmpty,
|
|
|
|
DataStore.shared().shouldDisplay(.showMeetingInMenubar) == false {
|
|
|
|
appDelegate.invalidateMenubarTimer(true)
|
|
|
|
}
|
|
|
|
|
|
|
|
if let appDelegate = NSApplication.shared.delegate as? AppDelegate {
|
|
|
|
appDelegate.setupMenubarTimer()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func setNewLabel(_ label: String, for dataObject: TimezoneData, at row: Int) {
|
|
|
|
let formattedValue = label.trimmingCharacters(in: NSCharacterSet.whitespacesAndNewlines)
|
|
|
|
|
|
|
|
if selectedTimeZones.count > row {
|
|
|
|
Logger.log(object: [
|
|
|
|
"Old Label": dataObject.customLabel ?? "Error",
|
|
|
|
"New Label": formattedValue,
|
|
|
|
],
|
|
|
|
for: "Custom Label Changed")
|
|
|
|
|
|
|
|
dataObject.setLabel(formattedValue)
|
|
|
|
|
|
|
|
insert(timezone: dataObject, at: row)
|
|
|
|
|
|
|
|
updateMenubarTitles()
|
|
|
|
} else {
|
|
|
|
Logger.log(object: [
|
|
|
|
"MethodName": "SetObjectValue",
|
|
|
|
"Selected Timezone Count": selectedTimeZones.count,
|
|
|
|
"Current Row": row,
|
|
|
|
],
|
|
|
|
for: "Error in selected row count")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func showAlertIfMoreThanOneTimezoneHasBeenAddedToTheMenubar() {
|
|
|
|
let isUITestRunning = ProcessInfo.processInfo.arguments.contains(CLUITestingLaunchArgument)
|
|
|
|
|
|
|
|
// If we have seen displayed the message before, abort!
|
|
|
|
let haveWeSeenThisMessageBefore = UserDefaults.standard.bool(forKey: CLLongStatusBarWarningMessage)
|
|
|
|
|
|
|
|
if haveWeSeenThisMessageBefore, !isUITestRunning {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the user is already using the compact mode, abort.
|
|
|
|
if DataStore.shared().shouldDisplay(.menubarCompactMode), !isUITestRunning {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Time to display the alert.
|
|
|
|
NSApplication.shared.activate(ignoringOtherApps: true)
|
|
|
|
|
|
|
|
let infoText = """
|
|
|
|
Multiple timezones occupy space and if macOS determines Clocker is occupying too much space, it'll hide Clocker entirely!
|
|
|
|
Enable Menubar Compact Mode to fit in more timezones in less space.
|
|
|
|
"""
|
|
|
|
|
|
|
|
let alert = NSAlert()
|
|
|
|
alert.showsSuppressionButton = true
|
|
|
|
alert.messageText = "More than one location added to the menubar 😅"
|
|
|
|
alert.informativeText = infoText
|
|
|
|
alert.addButton(withTitle: "Enable Compact Mode")
|
|
|
|
alert.addButton(withTitle: "Cancel")
|
|
|
|
|
|
|
|
let response = alert.runModal()
|
|
|
|
|
|
|
|
if response.rawValue == 1000 {
|
|
|
|
OperationQueue.main.addOperation {
|
|
|
|
UserDefaults.standard.set(0, forKey: CLMenubarCompactMode)
|
|
|
|
|
|
|
|
if alert.suppressionButton?.state == NSControl.StateValue.on {
|
|
|
|
UserDefaults.standard.set(true, forKey: CLLongStatusBarWarningMessage)
|
|
|
|
}
|
|
|
|
|
|
|
|
self.updateStatusBarAppearance()
|
|
|
|
|
|
|
|
Logger.log(object: ["Context": ">1 Menubar Timezone in Preferences"], for: "Switched to Compact Mode")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func tableView(_: NSTableView, writeRowsWith rowIndexes: IndexSet, to pboard: NSPasteboard) -> Bool {
|
|
|
|
let data = NSKeyedArchiver.archivedData(withRootObject: rowIndexes)
|
|
|
|
|
|
|
|
pboard.declareTypes([.dragSession], owner: self)
|
|
|
|
pboard.setData(data, forType: .dragSession)
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
func tableView(_: NSTableView, acceptDrop info: NSDraggingInfo, row: Int, dropOperation _: NSTableView.DropOperation) -> Bool {
|
|
|
|
var newOrder = selectedTimeZones
|
|
|
|
|
|
|
|
var destination = row
|
|
|
|
|
|
|
|
if row == newOrder.count {
|
|
|
|
destination -= 1
|
|
|
|
}
|
|
|
|
|
|
|
|
let pBoard = info.draggingPasteboard
|
|
|
|
|
|
|
|
guard let data = pBoard.data(forType: .dragSession) else {
|
|
|
|
assertionFailure("Data was unexpectedly nil")
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
guard let rowIndexes = NSKeyedUnarchiver.unarchiveObject(with: data) as? IndexSet, let first = rowIndexes.first else {
|
|
|
|
assertionFailure("Row was unexpectedly nil")
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
let currentObject = newOrder[first]
|
|
|
|
|
|
|
|
newOrder.remove(at: first)
|
|
|
|
|
|
|
|
newOrder.insert(currentObject, at: destination)
|
|
|
|
|
|
|
|
DataStore.shared().setTimezones(newOrder)
|
|
|
|
|
|
|
|
timezoneTableView.reloadData()
|
|
|
|
|
|
|
|
refreshMainTable()
|
|
|
|
|
|
|
|
timezoneTableView.deselectRow(timezoneTableView.selectedRow)
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
func tableView(_: NSTableView, validateDrop _: NSDraggingInfo, proposedRow _: Int, proposedDropOperation _: NSTableView.DropOperation) -> NSDragOperation {
|
|
|
|
return .every
|
|
|
|
}
|
|
|
|
|
|
|
|
func tableViewSelectionDidChange(_: Notification) {
|
|
|
|
deleteButton.isEnabled = !(timezoneTableView.selectedRow == -1)
|
|
|
|
}
|
|
|
|
|
|
|
|
func tableView(_ tableView: NSTableView, didClick tableColumn: NSTableColumn) {
|
|
|
|
if tableColumn.identifier.rawValue == "favouriteTimezone" {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if tableView == timezoneTableView {
|
|
|
|
let sortedTimezones = selectedTimeZones.sorted { (obj1, obj2) -> Bool in
|
|
|
|
|
|
|
|
guard let object1 = TimezoneData.customObject(from: obj1),
|
|
|
|
let object2 = TimezoneData.customObject(from: obj2) else {
|
|
|
|
assertionFailure("Data was unexpectedly nil")
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
if tableColumn.identifier.rawValue == "formattedAddress" {
|
|
|
|
return arePlacesSortedInAscendingOrder ?
|
|
|
|
object1.formattedAddress! > object2.formattedAddress! :
|
|
|
|
object1.formattedAddress! < object2.formattedAddress!
|
|
|
|
} else {
|
|
|
|
return arePlacesSortedInAscendingOrder ?
|
|
|
|
object1.customLabel! > object2.customLabel! :
|
|
|
|
object1.customLabel! < object2.customLabel!
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let indicatorImage = arePlacesSortedInAscendingOrder ?
|
|
|
|
NSImage(named: NSImage.Name("NSDescendingSortIndicator"))! :
|
|
|
|
NSImage(named: NSImage.Name("NSAscendingSortIndicator"))!
|
|
|
|
|
|
|
|
timezoneTableView.setIndicatorImage(indicatorImage, in: tableColumn)
|
|
|
|
|
|
|
|
arePlacesSortedInAscendingOrder.toggle()
|
|
|
|
|
|
|
|
DataStore.shared().setTimezones(sortedTimezones)
|
|
|
|
|
|
|
|
updateAfterSorting()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extension PreferencesViewController {
|
|
|
|
@objc private func search() {
|
|
|
|
let searchString = searchField.stringValue
|
|
|
|
|
|
|
|
if searchString.isEmpty {
|
|
|
|
dataTask?.cancel()
|
|
|
|
resetSearchView()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if dataTask?.state == .running {
|
|
|
|
dataTask?.cancel()
|
|
|
|
}
|
|
|
|
|
|
|
|
OperationQueue.main.addOperation {
|
|
|
|
if self.availableTimezoneTableView.isHidden {
|
|
|
|
self.availableTimezoneTableView.isHidden = false
|
|
|
|
}
|
|
|
|
|
|
|
|
self.placeholderLabel.isHidden = false
|
|
|
|
|
|
|
|
if NetworkManager.isConnected() == false {
|
|
|
|
self.placeholderLabel.placeholderString = PreferencesConstants.noInternetConnectivityError
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
self.isActivityInProgress = true
|
|
|
|
|
|
|
|
self.placeholderLabel.placeholderString = "Searching for \(searchString)"
|
|
|
|
|
|
|
|
self.dataTask = NetworkManager.task(with: self.generateSearchURL(),
|
|
|
|
completionHandler: { [weak self] response, error in
|
|
|
|
|
|
|
|
guard let self = self else { return }
|
|
|
|
|
|
|
|
OperationQueue.main.addOperation {
|
|
|
|
if let errorPresent = error {
|
|
|
|
self.presentError(errorPresent.localizedDescription)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
guard let data = response else {
|
|
|
|
assertionFailure("Data was unexpectedly nil")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
let searchResults = self.decode(from: data)
|
|
|
|
|
|
|
|
if searchResults?.status == "ZERO_RESULTS" {
|
|
|
|
self.placeholderLabel.placeholderString = "No results! 😔 Try entering the exact name."
|
|
|
|
self.isActivityInProgress = false
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
self.appendResultsToFilteredArray(searchResults!.results)
|
|
|
|
self.prepareUIForPresentingResults()
|
|
|
|
}
|
|
|
|
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func generateSearchURL() -> String {
|
|
|
|
let userPreferredLanguage = Locale.preferredLanguages.first ?? "en-US"
|
|
|
|
|
|
|
|
var searchString = searchField.stringValue
|
|
|
|
let words = searchString.components(separatedBy: CharacterSet.whitespacesAndNewlines)
|
|
|
|
searchString = words.joined(separator: CLEmptyString)
|
|
|
|
|
|
|
|
let url = "https://maps.googleapis.com/maps/api/geocode/json?address=\(searchString)&key=\(CLGeocodingKey)&language=\(userPreferredLanguage)"
|
|
|
|
return url
|
|
|
|
}
|
|
|
|
|
|
|
|
private func presentError(_ errorMessage: String) {
|
|
|
|
if errorMessage == PreferencesConstants.offlineErrorMessage {
|
|
|
|
placeholderLabel.placeholderString = PreferencesConstants.noInternetConnectivityError
|
|
|
|
} else {
|
|
|
|
placeholderLabel.placeholderString = PreferencesConstants.tryAgainMessage
|
|
|
|
}
|
|
|
|
|
|
|
|
isActivityInProgress = false
|
|
|
|
}
|
|
|
|
|
|
|
|
private func appendResultsToFilteredArray(_ results: [SearchResult.Result]) {
|
|
|
|
results.forEach {
|
|
|
|
let location = $0.geometry.location
|
|
|
|
let latitude = location.lat
|
|
|
|
let longitude = location.lng
|
|
|
|
let formattedAddress = $0.formattedAddress
|
|
|
|
|
|
|
|
let totalPackage = [
|
|
|
|
"latitude": latitude,
|
|
|
|
"longitude": longitude,
|
|
|
|
CLTimezoneName: formattedAddress,
|
|
|
|
CLCustomLabel: formattedAddress,
|
|
|
|
CLTimezoneID: CLEmptyString,
|
|
|
|
CLPlaceIdentifier: $0.placeId,
|
|
|
|
] as [String: Any]
|
|
|
|
|
|
|
|
self.filteredArray.append(TimezoneData(with: totalPackage))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func prepareUIForPresentingResults() {
|
|
|
|
placeholderLabel.placeholderString = CLEmptyString
|
|
|
|
isActivityInProgress = false
|
|
|
|
availableTimezoneTableView.reloadData()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Extracting this out for tests
|
|
|
|
private func decode(from data: Data) -> SearchResult? {
|
|
|
|
let jsonDecoder = JSONDecoder()
|
|
|
|
do {
|
|
|
|
let decodedObject = try jsonDecoder.decode(SearchResult.self, from: data)
|
|
|
|
return decodedObject
|
|
|
|
} catch {
|
|
|
|
print("decodedObject error: \n\(error)")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Extracting this out for tests
|
|
|
|
private func decodeTimezone(from data: Data) -> Timezone? {
|
|
|
|
let jsonDecoder = JSONDecoder()
|
|
|
|
do {
|
|
|
|
let decodedObject = try jsonDecoder.decode(Timezone.self, from: data)
|
|
|
|
return decodedObject
|
|
|
|
} catch {
|
|
|
|
print("decodedObject error: \n\(error)")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func resetSearchView() {
|
|
|
|
if dataTask?.state == .running {
|
|
|
|
dataTask?.cancel()
|
|
|
|
}
|
|
|
|
|
|
|
|
isActivityInProgress = false
|
|
|
|
placeholderLabel.placeholderString = CLEmptyString
|
|
|
|
}
|
|
|
|
|
|
|
|
private func getTimezone(for latitude: Double, and longitude: Double) {
|
|
|
|
if placeholderLabel.isHidden {
|
|
|
|
placeholderLabel.isHidden = false
|
|
|
|
}
|
|
|
|
|
|
|
|
if NetworkManager.isConnected() == false || ProcessInfo.processInfo.arguments.contains("mockTimezoneDown") {
|
|
|
|
resetStateAndShowDisconnectedMessage()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
searchField.placeholderString = "Fetching data might take some time!"
|
|
|
|
placeholderLabel.placeholderString = "Retrieving timezone data"
|
|
|
|
availableTimezoneTableView.isHidden = true
|
|
|
|
|
|
|
|
let tuple = "\(latitude),\(longitude)"
|
|
|
|
let timeStamp = Date().timeIntervalSince1970
|
|
|
|
let urlString = "https://maps.googleapis.com/maps/api/timezone/json?location=\(tuple)×tamp=\(timeStamp)&key=\(CLGeocodingKey)"
|
|
|
|
|
|
|
|
NetworkManager.task(with: urlString) { [weak self] response, error in
|
|
|
|
|
|
|
|
guard let self = self else { return }
|
|
|
|
|
|
|
|
OperationQueue.main.addOperation {
|
|
|
|
if self.handleEdgeCase(for: response) == true {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if error == nil, let json = response, let timezone = self.decodeTimezone(from: json) {
|
|
|
|
if self.availableTimezoneTableView.selectedRow >= 0, self.availableTimezoneTableView.selectedRow < self.filteredArray.count {
|
|
|
|
self.installTimezone(timezone)
|
|
|
|
}
|
|
|
|
self.updateViewState()
|
|
|
|
} else {
|
|
|
|
OperationQueue.main.addOperation {
|
|
|
|
if error?.localizedDescription == "The Internet connection appears to be offline." {
|
|
|
|
self.placeholderLabel.placeholderString = PreferencesConstants.noInternetConnectivityError
|
|
|
|
} else {
|
|
|
|
self.placeholderLabel.placeholderString = PreferencesConstants.tryAgainMessage
|
|
|
|
}
|
|
|
|
|
|
|
|
self.isActivityInProgress = false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func installTimezone(_ timezone: Timezone) {
|
|
|
|
guard let dataObject = self.filteredArray[self.availableTimezoneTableView.selectedRow] as? TimezoneData else {
|
|
|
|
assertionFailure("Data was unexpectedly nil")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
var filteredAddress = "Error"
|
|
|
|
|
|
|
|
if let address = dataObject.formattedAddress {
|
|
|
|
filteredAddress = address.filteredName()
|
|
|
|
}
|
|
|
|
|
|
|
|
let newTimeZone = [
|
|
|
|
CLTimezoneID: timezone.timeZoneId,
|
|
|
|
CLTimezoneName: filteredAddress,
|
|
|
|
CLPlaceIdentifier: dataObject.placeID!,
|
|
|
|
"latitude": dataObject.latitude!,
|
|
|
|
"longitude": dataObject.longitude!,
|
|
|
|
"nextUpdate": CLEmptyString,
|
|
|
|
CLCustomLabel: filteredAddress,
|
|
|
|
] as [String: Any]
|
|
|
|
|
|
|
|
let timezoneObject = TimezoneData(with: newTimeZone)
|
|
|
|
let operationsObject = TimezoneDataOperations(with: timezoneObject)
|
|
|
|
operationsObject.saveObject()
|
|
|
|
|
|
|
|
Logger.log(object: ["PlaceName": filteredAddress, "Timezone": timezone.timeZoneId], for: "Filtered Address")
|
|
|
|
}
|
|
|
|
|
|
|
|
private func resetStateAndShowDisconnectedMessage() {
|
|
|
|
OperationQueue.main.addOperation {
|
|
|
|
self.showMessage()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func showMessage() {
|
|
|
|
placeholderLabel.placeholderString = PreferencesConstants.noInternetConnectivityError
|
|
|
|
isActivityInProgress = false
|
|
|
|
filteredArray = []
|
|
|
|
availableTimezoneTableView.reloadData()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns true if there's an error.
|
|
|
|
private func handleEdgeCase(for response: Data?) -> Bool {
|
|
|
|
guard let json = response, let jsonUnserialized = try? JSONSerialization.jsonObject(with: json, options: .allowFragments), let unwrapped = jsonUnserialized as? [String: Any] else {
|
|
|
|
setErrorPlaceholders()
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
if let status = unwrapped["status"] as? String, status == "ZERO_RESULTS" {
|
|
|
|
setErrorPlaceholders()
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
private func setErrorPlaceholders() {
|
|
|
|
placeholderLabel.placeholderString = "No timezone found! Try entering an exact name."
|
|
|
|
searchField.placeholderString = "Enter a city, state or country name"
|
|
|
|
isActivityInProgress = false
|
|
|
|
}
|
|
|
|
|
|
|
|
private func updateViewState() {
|
|
|
|
filteredArray = []
|
|
|
|
availableTimezoneTableView.reloadData()
|
|
|
|
refreshTimezoneTableView()
|
|
|
|
refreshMainTable()
|
|
|
|
timezonePanel.close()
|
|
|
|
placeholderLabel.placeholderString = CLEmptyString
|
|
|
|
searchField.placeholderString = "Enter a city, state or country name"
|
|
|
|
availableTimezoneTableView.isHidden = false
|
|
|
|
isActivityInProgress = false
|
|
|
|
}
|
|
|
|
|
|
|
|
@IBAction func searchOptions(_: Any) {
|
|
|
|
placeholderLabel.placeholderString = CLEmptyString
|
|
|
|
placeholderLabel.isHidden = true
|
|
|
|
|
|
|
|
if searchCriteria.selectedSegment == 0 {
|
|
|
|
searchField.placeholderString = "Enter a city, state or country name"
|
|
|
|
columnName = "Place(s)"
|
|
|
|
abbreviation.isHidden = true
|
|
|
|
} else {
|
|
|
|
searchField.placeholderString = "Enter a timezone name"
|
|
|
|
columnName = "Timezone(s)"
|
|
|
|
abbreviation.isHidden = false
|
|
|
|
timezoneArray = []
|
|
|
|
timezoneArray.append("UTC")
|
|
|
|
timezoneArray.append("Anywhere on Earth")
|
|
|
|
timezoneArray.append(contentsOf: NSTimeZone.knownTimeZoneNames)
|
|
|
|
}
|
|
|
|
|
|
|
|
searchField.stringValue = CLEmptyString
|
|
|
|
availableTimezoneTableView.reloadData()
|
|
|
|
}
|
|
|
|
|
|
|
|
@IBAction func addTimeZone(_: NSButton) {
|
|
|
|
abbreviation.isHidden = true
|
|
|
|
filteredArray = []
|
|
|
|
searchCriteria.selectedSegment = 0
|
|
|
|
view.window?.beginSheet(timezonePanel,
|
|
|
|
completionHandler: nil)
|
|
|
|
}
|
|
|
|
|
|
|
|
@IBAction func addToFavorites(_: NSButton) {
|
|
|
|
isActivityInProgress = true
|
|
|
|
|
|
|
|
if availableTimezoneTableView.selectedRow == -1 {
|
|
|
|
messageLabel.stringValue = PreferencesConstants.noTimezoneSelectedErrorMessage
|
|
|
|
|
|
|
|
Timer.scheduledTimer(withTimeInterval: 5,
|
|
|
|
repeats: false) { _ in
|
|
|
|
OperationQueue.main.addOperation {
|
|
|
|
self.messageLabel.stringValue = CLEmptyString
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
isActivityInProgress = false
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if selectedTimeZones.count >= 100 {
|
|
|
|
messageLabel.stringValue = PreferencesConstants.maxTimezonesErrorMessage
|
|
|
|
Timer.scheduledTimer(withTimeInterval: 5,
|
|
|
|
repeats: false) { _ in
|
|
|
|
OperationQueue.main.addOperation {
|
|
|
|
self.messageLabel.stringValue = CLEmptyString
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
isActivityInProgress = false
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if searchCriteria.selectedSegment == 0 {
|
|
|
|
guard let dataObject = filteredArray[availableTimezoneTableView.selectedRow] as? TimezoneData else {
|
|
|
|
assertionFailure("Data was unexpectedly nil")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if messageLabel.stringValue.isEmpty {
|
|
|
|
searchField.stringValue = CLEmptyString
|
|
|
|
|
|
|
|
guard let latitude = dataObject.latitude, let longitude = dataObject.longitude else {
|
|
|
|
assertionFailure("Data was unexpectedly nil")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
getTimezone(for: latitude, and: longitude)
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
cleanupAfterInstallingTimezone()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func cleanupAfterInstallingTimezone() {
|
|
|
|
let data = TimezoneData()
|
|
|
|
data.setLabel(CLEmptyString)
|
|
|
|
|
|
|
|
if searchField.stringValue.isEmpty == false {
|
|
|
|
if timezoneFilteredArray.count <= availableTimezoneTableView.selectedRow {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
let currentSelection = timezoneFilteredArray[availableTimezoneTableView.selectedRow]
|
|
|
|
|
|
|
|
let metaInfo = metadata(for: currentSelection)
|
|
|
|
data.timezoneID = metaInfo.0
|
|
|
|
data.formattedAddress = metaInfo.1
|
|
|
|
|
|
|
|
} else {
|
|
|
|
let currentSelection = timezoneArray[availableTimezoneTableView.selectedRow]
|
|
|
|
|
|
|
|
let metaInfo = metadata(for: currentSelection)
|
|
|
|
data.timezoneID = metaInfo.0
|
|
|
|
data.formattedAddress = metaInfo.1
|
|
|
|
}
|
|
|
|
|
|
|
|
data.selectionType = .timezone
|
|
|
|
|
|
|
|
let operationObject = TimezoneDataOperations(with: data)
|
|
|
|
operationObject.saveObject()
|
|
|
|
|
|
|
|
timezoneFilteredArray = []
|
|
|
|
|
|
|
|
timezoneArray = []
|
|
|
|
|
|
|
|
availableTimezoneTableView.reloadData()
|
|
|
|
|
|
|
|
refreshTimezoneTableView()
|
|
|
|
|
|
|
|
refreshMainTable()
|
|
|
|
|
|
|
|
timezonePanel.close()
|
|
|
|
|
|
|
|
placeholderLabel.placeholderString = CLEmptyString
|
|
|
|
|
|
|
|
searchField.stringValue = CLEmptyString
|
|
|
|
|
|
|
|
searchField.placeholderString = "Enter a city, state or country name"
|
|
|
|
|
|
|
|
availableTimezoneTableView.isHidden = false
|
|
|
|
|
|
|
|
isActivityInProgress = false
|
|
|
|
}
|
|
|
|
|
|
|
|
private func metadata(for selection: String) -> (String, String) {
|
|
|
|
if selection == "Anywhere on Earth" {
|
|
|
|
return ("GMT-1200", selection)
|
|
|
|
} else if selection == "UTC" {
|
|
|
|
return ("GMT", selection)
|
|
|
|
} else {
|
|
|
|
return (selection, selection)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@IBAction func closePanel(_: NSButton) {
|
|
|
|
filteredArray = []
|
|
|
|
|
|
|
|
timezoneArray = []
|
|
|
|
|
|
|
|
searchCriteria.setSelected(true, forSegment: 0)
|
|
|
|
|
|
|
|
columnName = "Place(s)"
|
|
|
|
|
|
|
|
availableTimezoneTableView.reloadData()
|
|
|
|
|
|
|
|
searchField.stringValue = CLEmptyString
|
|
|
|
|
|
|
|
placeholderLabel.placeholderString = CLEmptyString
|
|
|
|
|
|
|
|
searchField.placeholderString = "Enter a city, state or country name"
|
|
|
|
|
|
|
|
timezonePanel.close()
|
|
|
|
|
|
|
|
isActivityInProgress = false
|
|
|
|
|
|
|
|
addTimezoneButton.state = .off
|
|
|
|
|
|
|
|
// The table might be hidden because of an early exit especially
|
|
|
|
// if we are not able to fetch an associated timezone
|
|
|
|
// For eg. Europe doesn't have an associated timezone
|
|
|
|
availableTimezoneTableView.isHidden = false
|
|
|
|
}
|
|
|
|
|
|
|
|
@IBAction func removeFromFavourites(_: NSButton) {
|
|
|
|
// If the user is editing a row, and decides to delete the row then we have a crash
|
|
|
|
if timezoneTableView.editedRow != -1 || timezoneTableView.editedColumn != -1 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if timezoneTableView.selectedRow == -1, selectedTimeZones.count <= timezoneTableView.selectedRow {
|
|
|
|
assertionFailure("Data was unexpectedly nil")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
let currentObject = selectedTimeZones[timezoneTableView.selectedRow]
|
|
|
|
guard let model = TimezoneData.customObject(from: currentObject) else {
|
|
|
|
assertionFailure("Data was unexpectedly nil")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if model.isFavourite == 1 {
|
|
|
|
removeFromMenubarFavourites(object: model)
|
|
|
|
}
|
|
|
|
|
|
|
|
var newDefaults = selectedTimeZones
|
|
|
|
|
|
|
|
let objectsToRemove = timezoneTableView.selectedRowIndexes.map { (index) -> Data in
|
|
|
|
selectedTimeZones[index]
|
|
|
|
}
|
|
|
|
|
|
|
|
newDefaults = newDefaults.filter { !objectsToRemove.contains($0) }
|
|
|
|
|
|
|
|
DataStore.shared().setTimezones(newDefaults)
|
|
|
|
|
|
|
|
timezoneTableView.reloadData()
|
|
|
|
|
|
|
|
refreshTimezoneTableView()
|
|
|
|
|
|
|
|
refreshMainTable()
|
|
|
|
|
|
|
|
if selectedTimeZones.isEmpty {
|
|
|
|
UserDefaults.standard.set(nil, forKey: CLMenubarFavorites)
|
|
|
|
}
|
|
|
|
|
|
|
|
updateStatusBarAppearance()
|
|
|
|
|
|
|
|
updateStatusItem()
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: This probably does not need to be used
|
|
|
|
private func updateStatusItem() {
|
|
|
|
guard let statusItem = (NSApplication.shared.delegate as? AppDelegate)?.statusItemForPanel() else {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
statusItem.performTimerWork()
|
|
|
|
}
|
|
|
|
|
|
|
|
private func updateStatusBarAppearance() {
|
|
|
|
guard let statusItem = (NSApplication.shared.delegate as? AppDelegate)?.statusItemForPanel() else {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
statusItem.setupStatusItem()
|
|
|
|
}
|
|
|
|
|
|
|
|
private func removeFromMenubarFavourites(object: TimezoneData?) {
|
|
|
|
guard let model = object else {
|
|
|
|
assertionFailure("Data was unexpectedly nil")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if model.isFavourite == 1 {
|
|
|
|
if let menubarTitles = DataStore.shared().retrieve(key: CLMenubarFavorites) as? [Data] {
|
|
|
|
let updated = menubarTitles.filter { (data) -> Bool in
|
|
|
|
let current = TimezoneData.customObject(from: data)
|
|
|
|
return current != model
|
|
|
|
}
|
|
|
|
|
|
|
|
UserDefaults.standard.set(updated, forKey: CLMenubarFavorites)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@IBAction func filterTimezoneArray(_: Any?) {
|
|
|
|
let lowercasedSearchString = searchField.stringValue.lowercased()
|
|
|
|
timezoneFilteredArray = timezoneArray.filter { $0.lowercased().contains(lowercasedSearchString) }
|
|
|
|
availableTimezoneTableView.reloadData()
|
|
|
|
}
|
|
|
|
|
|
|
|
@IBAction func filterArray(_ sender: Any?) {
|
|
|
|
if searchCriteria.selectedSegment == 1 {
|
|
|
|
filterTimezoneArray(sender)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
messageLabel.stringValue = CLEmptyString
|
|
|
|
|
|
|
|
filteredArray = []
|
|
|
|
|
|
|
|
if searchField.stringValue.count > 50 {
|
|
|
|
isActivityInProgress = false
|
|
|
|
messageLabel.stringValue = PreferencesConstants.maxCharactersAllowed
|
|
|
|
Timer.scheduledTimer(withTimeInterval: 5,
|
|
|
|
repeats: false) { _ in
|
|
|
|
OperationQueue.main.addOperation {
|
|
|
|
self.messageLabel.stringValue = CLEmptyString
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if searchField.stringValue.isEmpty == false {
|
|
|
|
dataTask?.cancel()
|
|
|
|
NSObject.cancelPreviousPerformRequests(withTarget: self)
|
|
|
|
perform(#selector(search), with: nil, afterDelay: 0.5)
|
|
|
|
} else {
|
|
|
|
resetSearchView()
|
|
|
|
}
|
|
|
|
|
|
|
|
availableTimezoneTableView.reloadData()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extension PreferencesViewController {
|
|
|
|
@IBAction func loginPreferenceChanged(_ sender: NSButton) {
|
|
|
|
if !SMLoginItemSetEnabled("com.abhishek.ClockerHelper" as CFString, sender.state == .on) {
|
|
|
|
Logger.log(object: ["Successful": "NO"], for: "Start Clocker Login")
|
|
|
|
addClockerToLoginItemsManually()
|
|
|
|
} else {
|
|
|
|
Logger.log(object: ["Successful": "YES"], for: "Start Clocker Login")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func addClockerToLoginItemsManually() {
|
|
|
|
NSApplication.shared.activate(ignoringOtherApps: true)
|
|
|
|
|
|
|
|
let alert = NSAlert()
|
|
|
|
alert.messageText = "Clocker is unable to set to start at login. 😅"
|
|
|
|
alert.informativeText = "You can manually set it to start at startup by adding Clocker to your login items."
|
|
|
|
alert.addButton(withTitle: "Add Manually")
|
|
|
|
alert.addButton(withTitle: "Cancel")
|
|
|
|
|
|
|
|
let response = alert.runModal()
|
|
|
|
if response.rawValue == 1000 {
|
|
|
|
OperationQueue.main.addOperation {
|
|
|
|
let prefPane = "/System/Library/PreferencePanes/Accounts.prefPane"
|
|
|
|
NSWorkspace.shared.openFile(prefPane)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Sorting
|
|
|
|
extension PreferencesViewController {
|
|
|
|
@IBAction func sortOptions(_: NSButton) {
|
|
|
|
additionalSortOptions.isHidden.toggle()
|
|
|
|
}
|
|
|
|
|
|
|
|
@IBAction func sortByTime(_ sender: NSButton) {
|
|
|
|
let sortedByTime = selectedTimeZones.sorted { (obj1, obj2) -> Bool in
|
|
|
|
|
|
|
|
let system = NSTimeZone.system
|
|
|
|
|
|
|
|
guard let object1 = TimezoneData.customObject(from: obj1),
|
|
|
|
let object2 = TimezoneData.customObject(from: obj2) else {
|
|
|
|
assertionFailure("Data was unexpectedly nil")
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
let timezone1 = NSTimeZone(name: object1.timezoneID!)
|
|
|
|
let timezone2 = NSTimeZone(name: object2.timezoneID!)
|
|
|
|
|
|
|
|
let difference1 = system.secondsFromGMT() - timezone1!.secondsFromGMT
|
|
|
|
let difference2 = system.secondsFromGMT() - timezone2!.secondsFromGMT
|
|
|
|
|
|
|
|
return arePlacesSortedInAscendingTimezoneOrder ? difference1 > difference2 : difference1 < difference2
|
|
|
|
}
|
|
|
|
|
|
|
|
sender.image = arePlacesSortedInAscendingTimezoneOrder ? NSImage(named: NSImage.Name("NSDescendingSortIndicator"))! : NSImage(named: NSImage.Name("NSAscendingSortIndicator"))!
|
|
|
|
|
|
|
|
arePlacesSortedInAscendingTimezoneOrder.toggle()
|
|
|
|
|
|
|
|
DataStore.shared().setTimezones(sortedByTime)
|
|
|
|
|
|
|
|
updateAfterSorting()
|
|
|
|
}
|
|
|
|
|
|
|
|
@IBAction func sortByLabel(_ sender: NSButton) {
|
|
|
|
let sortedLabels = selectedTimeZones.sorted { (obj1, obj2) -> Bool in
|
|
|
|
|
|
|
|
guard let object1 = TimezoneData.customObject(from: obj1),
|
|
|
|
let object2 = TimezoneData.customObject(from: obj2) else {
|
|
|
|
assertionFailure("Data was unexpectedly nil")
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return isLabelOptionSelected ? object1.customLabel! > object2.customLabel! : object1.customLabel! < object2.customLabel!
|
|
|
|
}
|
|
|
|
|
|
|
|
sender.image = isLabelOptionSelected ? NSImage(named: NSImage.Name("NSDescendingSortIndicator"))! : NSImage(named: NSImage.Name("NSAscendingSortIndicator"))!
|
|
|
|
|
|
|
|
isLabelOptionSelected.toggle()
|
|
|
|
|
|
|
|
DataStore.shared().setTimezones(sortedLabels)
|
|
|
|
|
|
|
|
updateAfterSorting()
|
|
|
|
}
|
|
|
|
|
|
|
|
@IBAction func sortByFormattedAddress(_ sender: NSButton) {
|
|
|
|
let sortedByAddress = selectedTimeZones.sorted { (obj1, obj2) -> Bool in
|
|
|
|
|
|
|
|
guard let object1 = TimezoneData.customObject(from: obj1),
|
|
|
|
let object2 = TimezoneData.customObject(from: obj2) else {
|
|
|
|
assertionFailure("Data was unexpectedly nil")
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return isTimezoneNameSortOptionSelected ? object1.formattedAddress! > object2.formattedAddress! : object1.formattedAddress! < object2.formattedAddress!
|
|
|
|
}
|
|
|
|
|
|
|
|
sender.image = isTimezoneNameSortOptionSelected ? NSImage(named: NSImage.Name("NSDescendingSortIndicator"))! : NSImage(named: NSImage.Name("NSAscendingSortIndicator"))!
|
|
|
|
|
|
|
|
isTimezoneNameSortOptionSelected.toggle()
|
|
|
|
|
|
|
|
DataStore.shared().setTimezones(sortedByAddress)
|
|
|
|
|
|
|
|
updateAfterSorting()
|
|
|
|
}
|
|
|
|
|
|
|
|
private func updateAfterSorting() {
|
|
|
|
let newDefaults = selectedTimeZones
|
|
|
|
|
|
|
|
DataStore.shared().setTimezones(newDefaults)
|
|
|
|
|
|
|
|
refreshTimezoneTableView()
|
|
|
|
refreshMainTable()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extension PreferencesViewController: SRRecorderControlDelegate {}
|
|
|
|
|
|
|
|
// Helpers
|
|
|
|
extension PreferencesViewController {
|
|
|
|
private func numberOfSearchResults() -> Int {
|
|
|
|
if searchCriteria.selectedSegment == 0 {
|
|
|
|
return filteredArray.count
|
|
|
|
}
|
|
|
|
|
|
|
|
if searchField.stringValue.isEmpty == false {
|
|
|
|
return timezoneFilteredArray.count
|
|
|
|
}
|
|
|
|
|
|
|
|
return timezoneArray.count
|
|
|
|
}
|
|
|
|
|
|
|
|
private func insert(timezone: TimezoneData, at index: Int) {
|
|
|
|
let encodedObject = NSKeyedArchiver.archivedData(withRootObject: timezone)
|
|
|
|
var newDefaults = selectedTimeZones
|
|
|
|
newDefaults[index] = encodedObject
|
|
|
|
DataStore.shared().setTimezones(newDefaults)
|
|
|
|
}
|
|
|
|
}
|