Browse Source

Add timezones to search results during Onboarding!

pull/92/head
Abhishek 4 years ago
parent
commit
54a55c15d4
  1. 157
      Clocker/Onboarding/OnboardingSearchController.swift
  2. 37
      Clocker/Preferences/General/PreferencesViewController.swift
  3. 41
      Clocker/Preferences/General/SearchDataSource.swift

157
Clocker/Onboarding/OnboardingSearchController.swift

@ -12,7 +12,7 @@ class OnboardingSearchController: NSViewController {
@IBOutlet private var accessoryLabel: NSTextField! @IBOutlet private var accessoryLabel: NSTextField!
@IBOutlet var undoButton: NSButton! @IBOutlet var undoButton: NSButton!
private var results: [TimezoneData] = [] private var searchResultsDataSource: SearchDataSource!
private var dataTask: URLSessionDataTask? = .none private var dataTask: URLSessionDataTask? = .none
private var themeDidChangeNotification: NSObjectProtocol? private var themeDidChangeNotification: NSObjectProtocol?
@ -31,6 +31,9 @@ class OnboardingSearchController: NSViewController {
view.wantsLayer = true view.wantsLayer = true
searchResultsDataSource = SearchDataSource(with: searchBar, location: .onboarding)
resultsTableView.isHidden = true
resultsTableView.delegate = self resultsTableView.delegate = self
resultsTableView.setAccessibility("ResultsTableView") resultsTableView.setAccessibility("ResultsTableView")
resultsTableView.dataSource = self resultsTableView.dataSource = self
@ -68,23 +71,68 @@ class OnboardingSearchController: NSViewController {
@objc func doubleClickAction(_: NSTableView?) { @objc func doubleClickAction(_: NSTableView?) {
[accessoryLabel].forEach { $0?.isHidden = false } [accessoryLabel].forEach { $0?.isHidden = false }
if resultsTableView.selectedRow >= 0, resultsTableView.selectedRow < results.count { if resultsTableView.selectedRow >= 0, resultsTableView.selectedRow < searchResultsDataSource.resultsCount() {
let selectedTimezone = results[resultsTableView.selectedRow] let selectedType = searchResultsDataSource.placeForRow(resultsTableView.selectedRow)
switch selectedType {
case .city:
let filteredGoogleResult = searchResultsDataSource.retrieveFilteredResultFromGoogleAPI(resultsTableView.selectedRow)
addTimezoneToDefaults(filteredGoogleResult!)
return
case .timezone:
cleanupAfterInstallingTimezone()
return
}
}
}
private func cleanupAfterInstallingTimezone() {
let data = TimezoneData()
data.setLabel(CLEmptyString)
let currentSelection = searchResultsDataSource.retrieveSelectedTimezone(resultsTableView.selectedRow)
let metaInfo = metadata(for: currentSelection)
data.timezoneID = metaInfo.0.name
data.formattedAddress = metaInfo.1.formattedName
data.selectionType = .timezone
data.isSystemTimezone = metaInfo.0.name == NSTimeZone.system.identifier
let operationObject = TimezoneDataOperations(with: data)
operationObject.saveObject()
searchResultsDataSource.cleanupFilterArray()
searchResultsDataSource.timezoneFilteredArray = []
searchResultsDataSource.calculateChangesets()
searchBar.stringValue = CLEmptyString
addTimezoneToDefaults(selectedTimezone) accessoryLabel.stringValue = "Added \(metaInfo.1.formattedName)."
undoButton.isHidden = false
setupLabelHidingTimer()
resultsTableView.reloadData()
resultsTableView.isHidden = true
}
private func metadata(for selection: TimezoneMetadata) -> (NSTimeZone, TimezoneMetadata) {
if selection.formattedName == "Anywhere on Earth" {
return (NSTimeZone(name: "GMT-1200")!, selection)
} else if selection.formattedName == "UTC" {
return (NSTimeZone(name: "GMT")!, selection)
} else {
return (selection.timezone, selection)
} }
} }
private func addTimezoneToDefaults(_ timezone: TimezoneData) { private func setupLabelHidingTimer() {
func setupLabelHidingTimer() {
Timer.scheduledTimer(withTimeInterval: 5, Timer.scheduledTimer(withTimeInterval: 5,
repeats: false) { _ in repeats: false) { _ in
OperationQueue.main.addOperation { OperationQueue.main.addOperation {
self.accessoryLabel.stringValue = CLEmptyString self.setInfoLabel(CLEmptyString)
} }
} }
} }
private func addTimezoneToDefaults(_ timezone: TimezoneData) {
if resultsTableView.selectedRow == -1 { if resultsTableView.selectedRow == -1 {
setInfoLabel(PreferencesConstants.noTimezoneSelectedErrorMessage) setInfoLabel(PreferencesConstants.noTimezoneSelectedErrorMessage)
setupLabelHidingTimer() setupLabelHidingTimer()
@ -131,22 +179,10 @@ class OnboardingSearchController: NSViewController {
return false return false
} }
// 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 {
Logger.info("decodedObject error: \n\(error)")
return nil
}
}
private func fetchTimezone(for latitude: Double, and longitude: Double, _ dataObject: TimezoneData) { private func fetchTimezone(for latitude: Double, and longitude: Double, _ dataObject: TimezoneData) {
if NetworkManager.isConnected() == false || ProcessInfo.processInfo.arguments.contains("mockTimezoneDown") { if NetworkManager.isConnected() == false || ProcessInfo.processInfo.arguments.contains("mockTimezoneDown") {
setInfoLabel(PreferencesConstants.noInternetConnectivityError) setInfoLabel(PreferencesConstants.noInternetConnectivityError)
results = [] searchResultsDataSource.cleanupFilterArray()
resultsTableView.reloadData() resultsTableView.reloadData()
return return
} }
@ -166,8 +202,8 @@ class OnboardingSearchController: NSViewController {
return return
} }
if error == nil, let json = response, let response = self.decodeTimezone(from: json) { if error == nil, let json = response, let response = json.decodeTimezone() {
if self.resultsTableView.selectedRow >= 0, self.resultsTableView.selectedRow < self.results.count { if self.resultsTableView.selectedRow >= 0, self.resultsTableView.selectedRow < self.searchResultsDataSource.resultsCount() {
var filteredAddress = "Error" var filteredAddress = "Error"
if let address = dataObject.formattedAddress { if let address = dataObject.formattedAddress {
@ -250,11 +286,6 @@ class OnboardingSearchController: NSViewController {
accessoryLabel.isHidden = false accessoryLabel.isHidden = false
if NetworkManager.isConnected() == false {
setInfoLabel(PreferencesConstants.noInternetConnectivityError)
return
}
NSObject.cancelPreviousPerformRequests(withTarget: self) NSObject.cancelPreviousPerformRequests(withTarget: self)
perform(#selector(OnboardingSearchController.actualSearch), with: nil, afterDelay: 0.2) perform(#selector(OnboardingSearchController.actualSearch), with: nil, afterDelay: 0.2)
} }
@ -268,6 +299,7 @@ class OnboardingSearchController: NSViewController {
@objc func actualSearch() { @objc func actualSearch() {
func setupForError() { func setupForError() {
searchResultsDataSource.calculateChangesets()
resultsTableView.isHidden = true resultsTableView.isHidden = true
} }
@ -300,21 +332,28 @@ class OnboardingSearchController: NSViewController {
return return
} }
self.results = [] self.searchResultsDataSource.cleanupFilterArray()
self.searchResultsDataSource.timezoneFilteredArray = []
if let errorPresent = error { if let errorPresent = error {
self.findLocalSearchResultsForTimezones()
if self.searchResultsDataSource.timezoneFilteredArray.count == 0 {
self.presentErrorMessage(errorPresent.localizedDescription) self.presentErrorMessage(errorPresent.localizedDescription)
setupForError() setupForError()
return return
} }
self.prepareUIForPresentingResults()
return
}
guard let data = response else { guard let data = response else {
self.setInfoLabel(PreferencesConstants.tryAgainMessage) self.setInfoLabel(PreferencesConstants.tryAgainMessage)
setupForError() setupForError()
return return
} }
let searchResults = self.decode(from: data) let searchResults = data.decode()
if searchResults?.status == "ZERO_RESULTS" { if searchResults?.status == "ZERO_RESULTS" {
self.setInfoLabel("No results! 😔 Try entering the exact name.") self.setInfoLabel("No results! 😔 Try entering the exact name.")
@ -323,10 +362,8 @@ class OnboardingSearchController: NSViewController {
} }
self.appendResultsToFilteredArray(searchResults!.results) self.appendResultsToFilteredArray(searchResults!.results)
self.findLocalSearchResultsForTimezones()
self.setInfoLabel(CLEmptyString) self.prepareUIForPresentingResults()
self.resultsTableView.reloadData()
} }
}) })
} }
@ -339,12 +376,25 @@ class OnboardingSearchController: NSViewController {
} }
} }
private func findLocalSearchResultsForTimezones() {
let lowercasedSearchString = searchBar.stringValue.lowercased()
searchResultsDataSource.searchTimezones(lowercasedSearchString)
}
private func prepareUIForPresentingResults() {
setInfoLabel(CLEmptyString)
if searchResultsDataSource.calculateChangesets() {
resultsTableView.isHidden = false
resultsTableView.reloadData()
}
}
private func appendResultsToFilteredArray(_ results: [SearchResult.Result]) { private func appendResultsToFilteredArray(_ results: [SearchResult.Result]) {
results.forEach { let finalTimezones: [TimezoneData] = results.map { (result) -> TimezoneData in
let location = $0.geometry.location let location = result.geometry.location
let latitude = location.lat let latitude = location.lat
let longitude = location.lng let longitude = location.lng
let formattedAddress = $0.formattedAddress let formattedAddress = result.formattedAddress
let totalPackage = [ let totalPackage = [
"latitude": latitude, "latitude": latitude,
@ -352,27 +402,17 @@ class OnboardingSearchController: NSViewController {
CLTimezoneName: formattedAddress, CLTimezoneName: formattedAddress,
CLCustomLabel: formattedAddress, CLCustomLabel: formattedAddress,
CLTimezoneID: CLEmptyString, CLTimezoneID: CLEmptyString,
CLPlaceIdentifier: $0.placeId, CLPlaceIdentifier: result.placeId,
] as [String: Any] ] as [String: Any]
self.results.append(TimezoneData(with: totalPackage)) return TimezoneData(with: totalPackage)
}
} }
// Extracting this out for tests searchResultsDataSource.setFilteredArrayValue(finalTimezones)
private func decode(from data: Data) -> SearchResult? {
let jsonDecoder = JSONDecoder()
do {
let decodedObject = try jsonDecoder.decode(SearchResult.self, from: data)
return decodedObject
} catch {
Logger.info("decodedObject error: \n\(error)")
return nil
}
} }
private func resetSearchView() { private func resetSearchView() {
results = [] searchResultsDataSource.cleanupFilterArray()
resultsTableView.reloadData() resultsTableView.reloadData()
searchBar.stringValue = CLEmptyString searchBar.stringValue = CLEmptyString
searchBar.placeholderString = "Press Enter to Search" searchBar.placeholderString = "Press Enter to Search"
@ -386,13 +426,18 @@ class OnboardingSearchController: NSViewController {
extension OnboardingSearchController: NSTableViewDataSource { extension OnboardingSearchController: NSTableViewDataSource {
func numberOfRows(in _: NSTableView) -> Int { func numberOfRows(in _: NSTableView) -> Int {
return results.count return searchResultsDataSource.resultsCount()
} }
func tableView(_ tableView: NSTableView, viewFor _: NSTableColumn?, row: Int) -> NSView? { func tableView(_ tableView: NSTableView, viewFor _: NSTableColumn?, row: Int) -> NSView? {
if let result = tableView.makeView(withIdentifier: NSUserInterfaceItemIdentifier(rawValue: "resultCellView"), owner: self) as? ResultTableViewCell, row >= 0, row < results.count { if let result = tableView.makeView(withIdentifier: NSUserInterfaceItemIdentifier(rawValue: "resultCellView"), owner: self) as? ResultTableViewCell, row >= 0, row < searchResultsDataSource.resultsCount() {
let currentTimezone = results[row] let currentSelection = searchResultsDataSource.retrieveResult(row)
result.result.stringValue = " \(currentTimezone.formattedAddress ?? "Place Name")" if let timezone = currentSelection as? TimezoneMetadata {
result.result.stringValue = " \(timezone.formattedName)"
} else if let location = currentSelection as? TimezoneData {
result.result.stringValue = " \(location.formattedAddress ?? "Place Name")"
}
result.result.textColor = Themer.shared().mainTextColor() result.result.textColor = Themer.shared().mainTextColor()
return result return result
} }
@ -403,7 +448,7 @@ extension OnboardingSearchController: NSTableViewDataSource {
extension OnboardingSearchController: NSTableViewDelegate { extension OnboardingSearchController: NSTableViewDelegate {
func tableView(_: NSTableView, heightOfRow row: Int) -> CGFloat { func tableView(_: NSTableView, heightOfRow row: Int) -> CGFloat {
if row == 0, results.isEmpty { if row == 0, searchResultsDataSource.resultsCount() == 0 {
return 30 return 30
} }
@ -411,7 +456,7 @@ extension OnboardingSearchController: NSTableViewDelegate {
} }
func tableView(_: NSTableView, shouldSelectRow row: Int) -> Bool { func tableView(_: NSTableView, shouldSelectRow row: Int) -> Bool {
return results.isEmpty ? row != 0 : true return searchResultsDataSource.resultsCount() == 0 ? row != 0 : true
} }
func tableView(_: NSTableView, rowViewForRow _: Int) -> NSTableRowView? { func tableView(_: NSTableView, rowViewForRow _: Int) -> NSTableRowView? {

37
Clocker/Preferences/General/PreferencesViewController.swift

@ -438,7 +438,7 @@ extension PreferencesViewController {
return return
} }
let searchResults = self.decode(from: data) let searchResults = data.decode()
if searchResults?.status == "ZERO_RESULTS" { if searchResults?.status == "ZERO_RESULTS" {
self.findLocalSearchResultsForTimezones() self.findLocalSearchResultsForTimezones()
@ -460,8 +460,6 @@ extension PreferencesViewController {
private func findLocalSearchResultsForTimezones() { private func findLocalSearchResultsForTimezones() {
let lowercasedSearchString = searchField.stringValue.lowercased() let lowercasedSearchString = searchField.stringValue.lowercased()
searchResultsDataSource.searchTimezones(lowercasedSearchString) searchResultsDataSource.searchTimezones(lowercasedSearchString)
Logger.info(searchResultsDataSource.timezoneFilteredArray.debugDescription)
} }
private func generateSearchURL() -> String { private func generateSearchURL() -> String {
@ -520,30 +518,6 @@ extension PreferencesViewController {
} }
} }
// 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 {
Logger.info("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 {
Logger.info("decodedObject error: \n\(error)")
return nil
}
}
private func resetSearchView() { private func resetSearchView() {
if dataTask?.state == .running { if dataTask?.state == .running {
dataTask?.cancel() dataTask?.cancel()
@ -576,7 +550,7 @@ extension PreferencesViewController {
return return
} }
if error == nil, let json = response, let timezone = strongSelf.decodeTimezone(from: json) { if error == nil, let json = response, let timezone = json.decodeTimezone() {
if strongSelf.availableTimezoneTableView.selectedRow >= 0 { if strongSelf.availableTimezoneTableView.selectedRow >= 0 {
strongSelf.installTimezone(timezone) strongSelf.installTimezone(timezone)
} }
@ -597,7 +571,7 @@ extension PreferencesViewController {
} }
private func installTimezone(_ timezone: Timezone) { private func installTimezone(_ timezone: Timezone) {
guard let dataObject = searchResultsDataSource.retrieveFilteredResult(availableTimezoneTableView.selectedRow) else { guard let dataObject = searchResultsDataSource.retrieveFilteredResultFromGoogleAPI(availableTimezoneTableView.selectedRow) else {
assertionFailure("Data was unexpectedly nil") assertionFailure("Data was unexpectedly nil")
return return
} }
@ -742,7 +716,7 @@ extension PreferencesViewController {
} }
private func cleanupAfterInstallingCity() { private func cleanupAfterInstallingCity() {
guard let dataObject = searchResultsDataSource.retrieveFilteredResult(availableTimezoneTableView.selectedRow) else { guard let dataObject = searchResultsDataSource.retrieveFilteredResultFromGoogleAPI(availableTimezoneTableView.selectedRow) else {
assertionFailure("Data was unexpectedly nil") assertionFailure("Data was unexpectedly nil")
return return
} }
@ -763,8 +737,7 @@ extension PreferencesViewController {
let data = TimezoneData() let data = TimezoneData()
data.setLabel(CLEmptyString) data.setLabel(CLEmptyString)
let currentSelection = searchResultsDataSource.retrieveSelectedTimezone(searchField.stringValue, let currentSelection = searchResultsDataSource.retrieveSelectedTimezone(availableTimezoneTableView.selectedRow)
availableTimezoneTableView.selectedRow)
let metaInfo = metadata(for: currentSelection) let metaInfo = metadata(for: currentSelection)
data.timezoneID = metaInfo.0.name data.timezoneID = metaInfo.0.name

41
Clocker/Preferences/General/SearchDataSource.swift

@ -8,6 +8,11 @@ enum RowType {
case timezone case timezone
} }
enum SearchLocation {
case onboarding
case preferences
}
struct TimezoneMetadata { struct TimezoneMetadata {
let timezone: NSTimeZone let timezone: NSTimeZone
let tags: Set<String> let tags: Set<String>
@ -18,6 +23,7 @@ struct TimezoneMetadata {
class SearchDataSource: NSObject { class SearchDataSource: NSObject {
private var searchField: NSSearchField! private var searchField: NSSearchField!
private var finalArray: [RowType] = [] private var finalArray: [RowType] = []
private var location: SearchLocation = .preferences
private var dataTask: URLSessionDataTask? = .none private var dataTask: URLSessionDataTask? = .none
private var timezoneMetadataDictionary: [String: [String]] = private var timezoneMetadataDictionary: [String: [String]] =
["GMT+5:30": ["india", "indian", "kolkata", "calcutta", "mumbai", "delhi", "hyderabad", "noida"], ["GMT+5:30": ["india", "indian", "kolkata", "calcutta", "mumbai", "delhi", "hyderabad", "noida"],
@ -27,13 +33,14 @@ class SearchDataSource: NSObject {
"EST": ["florida", "new york"], "EST": ["florida", "new york"],
"EDT": ["florida", "new york"]] "EDT": ["florida", "new york"]]
private var filteredArray: [TimezoneData] = [] private var filteredArray: [TimezoneData] = [] // Filtered results from the Google API
private var timezoneArray: [TimezoneMetadata] = [] private var timezoneArray: [TimezoneMetadata] = [] // All timezones
var timezoneFilteredArray: [TimezoneMetadata] = [] var timezoneFilteredArray: [TimezoneMetadata] = [] // Filtered timezones list based on search input
init(with searchField: NSSearchField) { init(with searchField: NSSearchField, location: SearchLocation = .preferences) {
super.init() super.init()
self.searchField = searchField self.searchField = searchField
self.location = location
setupTimezoneDatasource() setupTimezoneDatasource()
calculateChangesets() calculateChangesets()
} }
@ -50,10 +57,28 @@ class SearchDataSource: NSObject {
return finalArray[row] return finalArray[row]
} }
func retrieveFilteredResult(_ index: Int) -> TimezoneData? { // Returns result from finalArray based on the type i.e. city or timezone
func retrieveResult(_ row: Int) -> Any? {
let currentRowType = finalArray[row]
switch currentRowType {
case .timezone:
let datasource = searchField.stringValue.isEmpty ? timezoneArray : timezoneFilteredArray
return datasource[row % datasource.count]
case .city:
let timezoneData = filteredArray[row % filteredArray.count]
return timezoneData
}
}
func retrieveFilteredResultFromGoogleAPI(_ index: Int) -> TimezoneData? {
return filteredArray[index % filteredArray.count] return filteredArray[index % filteredArray.count]
} }
func resultsCount() -> Int {
return finalArray.count
}
private func setupTimezoneDatasource() { private func setupTimezoneDatasource() {
timezoneArray = [] timezoneArray = []
@ -103,7 +128,7 @@ class SearchDataSource: NSObject {
} }
} }
if searchField.stringValue.isEmpty { if searchField.stringValue.isEmpty, location == .preferences {
addTimezonesIfNeeded(timezoneArray) addTimezonesIfNeeded(timezoneArray)
} else { } else {
filteredArray.forEach { _ in filteredArray.forEach { _ in
@ -132,8 +157,8 @@ class SearchDataSource: NSObject {
} }
} }
func retrieveSelectedTimezone(_ searchString: String, _ selectedIndex: Int) -> TimezoneMetadata { func retrieveSelectedTimezone(_ selectedIndex: Int) -> TimezoneMetadata {
return searchString.isEmpty == false ? timezoneFilteredArray[selectedIndex % timezoneFilteredArray.count] : return searchField.stringValue.isEmpty == false ? timezoneFilteredArray[selectedIndex % timezoneFilteredArray.count] :
timezoneArray[selectedIndex] timezoneArray[selectedIndex]
} }
} }

Loading…
Cancel
Save